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");
16units::Length distance(
const ASN1_PREFIXED(ReferencePosition_t)& a,
const ASN1_PREFIXED(ReferencePosition_t)& b)
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)) {
23 GeodeticPosition geo_a {
24 GeoAngle { a.latitude * tenth_microdegree },
25 GeoAngle { a.longitude * tenth_microdegree }
27 GeodeticPosition geo_b {
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)) {
43 GeodeticPosition geo_a {
44 GeoAngle { a.latitude * tenth_microdegree },
45 GeoAngle { a.longitude * tenth_microdegree }
47 GeodeticPosition geo_b { lat, lon };
48 length = geonet::distance(geo_a, geo_b);
53bool is_available(
const ASN1_PREFIXED(ReferencePosition)& pos)
55 return pos.latitude != ASN1_PREFIXED(Latitude_unavailable) && pos.longitude != ASN1_PREFIXED(Longitude_unavailable);
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);