/* * Copyright (C) 2014 Jared Boone, ShareBrained Technology, Inc. * * This file is part of PortaPack. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This program 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 General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; see the file COPYING. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, * Boston, MA 02110-1301, USA. */ #ifndef __AIS_PACKET_H__ #define __AIS_PACKET_H__ #include "baseband_packet.hpp" #include "field_reader.hpp" #include #include #include namespace ais { struct DateTime { uint16_t year; uint8_t month; uint8_t day; uint8_t hour; uint8_t minute; uint8_t second; }; template struct LatLonBase { constexpr LatLonBase( ) : LatLonBase { raw_not_available } { } constexpr LatLonBase( const int32_t raw ) : raw_ { raw } { } constexpr LatLonBase( const LatLonBase& other ) : raw_ { other.raw_ } { } int32_t normalized() const { return static_cast(raw() << sign_extend_shift) / (1 << sign_extend_shift); } int32_t raw() const { return raw_; } bool is_not_available() const { return raw() == raw_not_available; } bool is_valid() const { return (normalized() >= raw_valid_min) && (normalized() <= raw_valid_max); } private: int32_t raw_; static constexpr size_t sign_extend_shift = 32 - FieldSize; static constexpr int32_t raw_not_available = NAValue; static constexpr int32_t raw_valid_min = -DegMax * 60 * 10000; static constexpr int32_t raw_valid_max = DegMax * 60 * 10000; }; using Latitude = LatLonBase<27, 90, 0x3412140>; using Longitude = LatLonBase<28, 180, 0x6791AC0>; using RateOfTurn = int8_t; using SpeedOverGround = uint16_t; using CourseOverGround = uint16_t; using TrueHeading = uint16_t; using MMSI = uint32_t; class Packet { public: constexpr Packet( const baseband::Packet& packet ) : packet_ { packet }, field_ { packet_ } { } size_t length() const; bool is_valid() const; Timestamp received_at() const; uint32_t message_id() const; MMSI user_id() const; MMSI source_id() const; uint32_t read(const size_t start_bit, const size_t length) const; std::string text(const size_t start_bit, const size_t character_count) const; DateTime datetime(const size_t start_bit) const; Latitude latitude(const size_t start_bit) const; Longitude longitude(const size_t start_bit) const; bool crc_ok() const; private: using Reader = FieldReader; using CRCReader = FieldReader; const baseband::Packet packet_; const Reader field_; const size_t fcs_length = 16; size_t data_and_fcs_length() const; size_t data_length() const; bool length_valid() const; }; } /* namespace ais */ #endif/*__AIS_PACKET_H__*/