1#include <vanetza/asn1/its/ReferencePosition.h>
2#include <vanetza/asn1/its/r2/ReferencePosition.h>
3#include <vanetza/facilities/detail/macros.ipp>
4#include <vanetza/geonet/areas.hpp>
5#include <vanetza/units/length.hpp>
13static_assert(Longitude_oneMicrodegreeEast == 10,
"Longitude is an integer number of tenth microdegrees");
14static_assert(Latitude_oneMicrodegreeNorth == 10,
"Latitude is an integer number of tenth microdegrees");
18 using geonet::GeodeticPosition;
19 using units::GeoAngle;
21 auto length = units::Length::from_value(std::numeric_limits<
double>::quiet_NaN());
22 if (is_available(a) && is_available(b)) {
24 GeoAngle { a.latitude * tenth_microdegree },
25 GeoAngle { a.longitude * tenth_microdegree }
28 GeoAngle { b.latitude * tenth_microdegree },
29 GeoAngle { b.longitude * tenth_microdegree }
31 length = geonet::distance(geo_a, geo_b);
36units::Length distance(
const ASN1_PREFIXED(ReferencePosition_t)& a, units::GeoAngle lat, units::GeoAngle lon)
38 using geonet::GeodeticPosition;
39 using units::GeoAngle;
41 auto length = units::Length::from_value(std::numeric_limits<
double>::quiet_NaN());
42 if (is_available(a)) {
44 GeoAngle { a.latitude * tenth_microdegree },
45 GeoAngle { a.longitude * tenth_microdegree }
48 length = geonet::distance(geo_a, geo_b);
58void copy(
const PositionFix& position,
ASN1_PREFIXED(ReferencePosition)& reference_position)
60 reference_position.longitude = round(position.longitude, tenth_microdegree);
61 reference_position.latitude = round(position.latitude, tenth_microdegree);
62 if (std::isfinite(position.confidence.semi_major.value())
63 && std::isfinite(position.confidence.semi_minor.value()))
65 if ((position.confidence.semi_major.value() * 100 <
ASN1_PREFIXED(SemiAxisLength_outOfRange))
66 && (position.confidence.semi_minor.value() * 100 <
ASN1_PREFIXED(SemiAxisLength_outOfRange))
67 && (position.confidence.orientation.value() * 10 <
ASN1_PREFIXED(HeadingValue_unavailable)))
69 reference_position.positionConfidenceEllipse.semiMajorConfidence = position.confidence.semi_major.value() * 100;
70 reference_position.positionConfidenceEllipse.semiMinorConfidence = position.confidence.semi_minor.value() * 100;
71 reference_position.positionConfidenceEllipse.semiMajorOrientation = (position.confidence.orientation.value()) * 10;
75 reference_position.positionConfidenceEllipse.semiMajorConfidence =
ASN1_PREFIXED(SemiAxisLength_outOfRange);
76 reference_position.positionConfidenceEllipse.semiMinorConfidence =
ASN1_PREFIXED(SemiAxisLength_outOfRange);
77 reference_position.positionConfidenceEllipse.semiMajorOrientation =
ASN1_PREFIXED(HeadingValue_unavailable);
82 reference_position.positionConfidenceEllipse.semiMajorConfidence =
ASN1_PREFIXED(SemiAxisLength_unavailable);
83 reference_position.positionConfidenceEllipse.semiMinorConfidence =
ASN1_PREFIXED(SemiAxisLength_unavailable);
84 reference_position.positionConfidenceEllipse.semiMajorOrientation =
ASN1_PREFIXED(HeadingValue_unavailable);
86 if (position.altitude) {
87 reference_position.altitude.altitudeValue = to_altitude_value(position.altitude->value());
88 reference_position.altitude.altitudeConfidence = to_altitude_confidence(position.altitude->confidence());
90 reference_position.altitude.altitudeValue =
ASN1_PREFIXED(AltitudeValue_unavailable);
91 reference_position.altitude.altitudeConfidence =
ASN1_PREFIXED(AltitudeConfidence_unavailable);
#define ASN1_PREFIXED(name)