mirror of https://github.com/ArduPilot/ardupilot
219 lines
6.4 KiB
C++
219 lines
6.4 KiB
C++
/*
|
|
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 3 of the License, 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. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "AP_GPS.h"
|
|
#include "GPS_Backend.h"
|
|
|
|
#define GPS_BACKEND_DEBUGGING 0
|
|
|
|
#if GPS_BACKEND_DEBUGGING
|
|
# define Debug(fmt, args ...) do {hal.console->printf("%s:%d: " fmt "\n", __FUNCTION__, __LINE__, ## args); hal.scheduler->delay(1); } while(0)
|
|
#else
|
|
# define Debug(fmt, args ...)
|
|
#endif
|
|
|
|
#include <GCS_MAVLink/GCS.h>
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
AP_GPS_Backend::AP_GPS_Backend(AP_GPS &_gps, AP_GPS::GPS_State &_state, AP_HAL::UARTDriver *_port) :
|
|
port(_port),
|
|
gps(_gps),
|
|
state(_state)
|
|
{
|
|
state.have_speed_accuracy = false;
|
|
state.have_horizontal_accuracy = false;
|
|
state.have_vertical_accuracy = false;
|
|
}
|
|
|
|
int32_t AP_GPS_Backend::swap_int32(int32_t v) const
|
|
{
|
|
const uint8_t *b = (const uint8_t *)&v;
|
|
union {
|
|
int32_t v;
|
|
uint8_t b[4];
|
|
} u;
|
|
|
|
u.b[0] = b[3];
|
|
u.b[1] = b[2];
|
|
u.b[2] = b[1];
|
|
u.b[3] = b[0];
|
|
|
|
return u.v;
|
|
}
|
|
|
|
int16_t AP_GPS_Backend::swap_int16(int16_t v) const
|
|
{
|
|
const uint8_t *b = (const uint8_t *)&v;
|
|
union {
|
|
int16_t v;
|
|
uint8_t b[2];
|
|
} u;
|
|
|
|
u.b[0] = b[1];
|
|
u.b[1] = b[0];
|
|
|
|
return u.v;
|
|
}
|
|
|
|
|
|
/**
|
|
fill in time_week_ms and time_week from BCD date and time components
|
|
assumes MTK19 millisecond form of bcd_time
|
|
*/
|
|
void AP_GPS_Backend::make_gps_time(uint32_t bcd_date, uint32_t bcd_milliseconds)
|
|
{
|
|
uint8_t year, mon, day, hour, min, sec;
|
|
uint16_t msec;
|
|
|
|
year = bcd_date % 100;
|
|
mon = (bcd_date / 100) % 100;
|
|
day = bcd_date / 10000;
|
|
|
|
uint32_t v = bcd_milliseconds;
|
|
msec = v % 1000; v /= 1000;
|
|
sec = v % 100; v /= 100;
|
|
min = v % 100; v /= 100;
|
|
hour = v % 100;
|
|
|
|
int8_t rmon = mon - 2;
|
|
if (0 >= rmon) {
|
|
rmon += 12;
|
|
year -= 1;
|
|
}
|
|
|
|
// get time in seconds since unix epoch
|
|
uint32_t ret = (year/4) - (GPS_LEAPSECONDS_MILLIS / 1000UL) + 367*rmon/12 + day;
|
|
ret += year*365 + 10501;
|
|
ret = ret*24 + hour;
|
|
ret = ret*60 + min;
|
|
ret = ret*60 + sec;
|
|
|
|
// convert to time since GPS epoch
|
|
ret -= 272764785UL;
|
|
|
|
// get GPS week and time
|
|
state.time_week = ret / AP_SEC_PER_WEEK;
|
|
state.time_week_ms = (ret % AP_SEC_PER_WEEK) * AP_MSEC_PER_SEC;
|
|
state.time_week_ms += msec;
|
|
}
|
|
|
|
/*
|
|
fill in 3D velocity for a GPS that doesn't give vertical velocity numbers
|
|
*/
|
|
void AP_GPS_Backend::fill_3d_velocity(void)
|
|
{
|
|
float gps_heading = radians(state.ground_course);
|
|
|
|
state.velocity.x = state.ground_speed * cosf(gps_heading);
|
|
state.velocity.y = state.ground_speed * sinf(gps_heading);
|
|
state.velocity.z = 0;
|
|
state.have_vertical_velocity = false;
|
|
}
|
|
|
|
void
|
|
AP_GPS_Backend::inject_data(const uint8_t *data, uint16_t len)
|
|
{
|
|
// not all backends have valid ports
|
|
if (port != nullptr) {
|
|
if (port->txspace() > len) {
|
|
port->write(data, len);
|
|
} else {
|
|
Debug("GPS %d: Not enough TXSPACE", state.instance + 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
void AP_GPS_Backend::_detection_message(char *buffer, const uint8_t buflen) const
|
|
{
|
|
const uint8_t instance = state.instance;
|
|
const struct AP_GPS::detect_state dstate = gps.detect_state[instance];
|
|
|
|
if (dstate.auto_detected_baud) {
|
|
hal.util->snprintf(buffer, buflen,
|
|
"GPS %d: detected as %s at %d baud",
|
|
instance + 1,
|
|
name(),
|
|
gps._baudrates[dstate.current_baud]);
|
|
} else {
|
|
hal.util->snprintf(buffer, buflen,
|
|
"GPS %d: specified as %s",
|
|
instance + 1,
|
|
name());
|
|
}
|
|
}
|
|
|
|
|
|
void AP_GPS_Backend::broadcast_gps_type() const
|
|
{
|
|
char buffer[MAVLINK_MSG_STATUSTEXT_FIELD_TEXT_LEN+1];
|
|
_detection_message(buffer, sizeof(buffer));
|
|
gcs().send_text(MAV_SEVERITY_INFO, buffer);
|
|
}
|
|
|
|
void AP_GPS_Backend::Write_DataFlash_Log_Startup_messages() const
|
|
{
|
|
char buffer[MAVLINK_MSG_STATUSTEXT_FIELD_TEXT_LEN+1];
|
|
_detection_message(buffer, sizeof(buffer));
|
|
DataFlash_Class::instance()->Log_Write_Message(buffer);
|
|
}
|
|
|
|
bool AP_GPS_Backend::should_df_log() const
|
|
{
|
|
return gps.should_df_log();
|
|
}
|
|
|
|
|
|
void AP_GPS_Backend::send_mavlink_gps_rtk(mavlink_channel_t chan)
|
|
{
|
|
const uint8_t instance = state.instance;
|
|
// send status
|
|
switch (instance) {
|
|
case 0:
|
|
mavlink_msg_gps_rtk_send(chan,
|
|
0, // Not implemented yet
|
|
0, // Not implemented yet
|
|
state.rtk_week_number,
|
|
state.rtk_time_week_ms,
|
|
0, // Not implemented yet
|
|
0, // Not implemented yet
|
|
state.rtk_num_sats,
|
|
state.rtk_baseline_coords_type,
|
|
state.rtk_baseline_x_mm,
|
|
state.rtk_baseline_y_mm,
|
|
state.rtk_baseline_z_mm,
|
|
state.rtk_accuracy,
|
|
state.rtk_iar_num_hypotheses);
|
|
break;
|
|
case 1:
|
|
mavlink_msg_gps2_rtk_send(chan,
|
|
0, // Not implemented yet
|
|
0, // Not implemented yet
|
|
state.rtk_week_number,
|
|
state.rtk_time_week_ms,
|
|
0, // Not implemented yet
|
|
0, // Not implemented yet
|
|
state.rtk_num_sats,
|
|
state.rtk_baseline_coords_type,
|
|
state.rtk_baseline_x_mm,
|
|
state.rtk_baseline_y_mm,
|
|
state.rtk_baseline_z_mm,
|
|
state.rtk_accuracy,
|
|
state.rtk_iar_num_hypotheses);
|
|
break;
|
|
}
|
|
}
|
|
|