/**************************************************************************** ** ** Copyright (C) 2016 Jolla Ltd. ** Contact: Aaron McCarthy ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the QtPositioning module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qnmeapositioninfosource_p.h" #include "qlocationutils_p.h" #include #include #include #include #include QT_BEGIN_NAMESPACE QNmeaRealTimeReader::QNmeaRealTimeReader(QNmeaPositionInfoSourcePrivate *sourcePrivate) : QNmeaReader(sourcePrivate) { } void QNmeaRealTimeReader::readAvailableData() { while (m_proxy->m_device->canReadLine()){ QGeoPositionInfo update; bool hasFix = false; char buf[1024]; qint64 size = m_proxy->m_device->readLine(buf, sizeof(buf)); if (m_proxy->parsePosInfoFromNmeaData(buf, size, &update, &hasFix)) m_proxy->notifyNewUpdate(&update, hasFix); } } //============================================================ QNmeaSimulatedReader::QNmeaSimulatedReader(QNmeaPositionInfoSourcePrivate *sourcePrivate) : QNmeaReader(sourcePrivate), m_currTimerId(-1), m_hasValidDateTime(false) { } QNmeaSimulatedReader::~QNmeaSimulatedReader() { if (m_currTimerId > 0) killTimer(m_currTimerId); } void QNmeaSimulatedReader::readAvailableData() { if (m_currTimerId > 0) // we are already reading return; if (!m_hasValidDateTime) { // first update Q_ASSERT(m_proxy->m_device && (m_proxy->m_device->openMode() & QIODevice::ReadOnly)); if (!setFirstDateTime()) { //m_proxy->notifyReachedEndOfFile(); qWarning("QNmeaPositionInfoSource: cannot find NMEA sentence with valid date & time"); return; } m_hasValidDateTime = true; simulatePendingUpdate(); } else { // previously read to EOF, but now new data has arrived processNextSentence(); } } bool QNmeaSimulatedReader::setFirstDateTime() { // find the first update with valid date and time QGeoPositionInfo update; bool hasFix = false; while (m_proxy->m_device->bytesAvailable() > 0) { char buf[1024]; qint64 size = m_proxy->m_device->readLine(buf, sizeof(buf)); if (size <= 0) continue; bool ok = m_proxy->parsePosInfoFromNmeaData(buf, size, &update, &hasFix); if (ok && update.timestamp().isValid()) { QPendingGeoPositionInfo pending; pending.info = update; pending.hasFix = hasFix; m_pendingUpdates.enqueue(pending); return true; } } return false; } void QNmeaSimulatedReader::simulatePendingUpdate() { if (m_pendingUpdates.size() > 0) { // will be dequeued in processNextSentence() QPendingGeoPositionInfo &pending = m_pendingUpdates.head(); m_proxy->notifyNewUpdate(&pending.info, pending.hasFix); } processNextSentence(); } void QNmeaSimulatedReader::timerEvent(QTimerEvent *event) { killTimer(event->timerId()); m_currTimerId = -1; simulatePendingUpdate(); } void QNmeaSimulatedReader::processNextSentence() { QGeoPositionInfo info; bool hasFix = false; int timeToNextUpdate = -1; QTime prevTime; if (m_pendingUpdates.size() > 0) prevTime = m_pendingUpdates.head().info.timestamp().time(); // find the next update with a valid time (as long as the time is valid, // we can calculate when the update should be emitted) while (m_proxy->m_device && m_proxy->m_device->bytesAvailable() > 0) { char buf[1024]; qint64 size = m_proxy->m_device->readLine(buf, sizeof(buf)); if (size <= 0) continue; if (m_proxy->parsePosInfoFromNmeaData(buf, size, &info, &hasFix)) { QTime time = info.timestamp().time(); if (time.isValid()) { if (!prevTime.isValid()) { timeToNextUpdate = 0; break; } timeToNextUpdate = prevTime.msecsTo(time); if (timeToNextUpdate >= 0) break; } else { timeToNextUpdate = 0; break; } } } if (timeToNextUpdate < 0) return; m_pendingUpdates.dequeue(); QPendingGeoPositionInfo pending; pending.info = info; pending.hasFix = hasFix; m_pendingUpdates.enqueue(pending); m_currTimerId = startTimer(timeToNextUpdate); } //============================================================ QNmeaPositionInfoSourcePrivate::QNmeaPositionInfoSourcePrivate(QNmeaPositionInfoSource *parent, QNmeaPositionInfoSource::UpdateMode updateMode) : QObject(parent), m_updateMode(updateMode), m_device(0), m_invokedStart(false), m_positionError(QGeoPositionInfoSource::UnknownSourceError), m_userEquivalentRangeError(qQNaN()), m_source(parent), m_nmeaReader(0), m_updateTimer(0), m_requestTimer(0), m_horizontalAccuracy(qQNaN()), m_verticalAccuracy(qQNaN()), m_noUpdateLastInterval(false), m_updateTimeoutSent(false), m_connectedReadyRead(false) { } QNmeaPositionInfoSourcePrivate::~QNmeaPositionInfoSourcePrivate() { delete m_nmeaReader; delete m_updateTimer; } bool QNmeaPositionInfoSourcePrivate::openSourceDevice() { if (!m_device) { qWarning("QNmeaPositionInfoSource: no QIODevice data source, call setDevice() first"); return false; } if (!m_device->isOpen() && !m_device->open(QIODevice::ReadOnly)) { qWarning("QNmeaPositionInfoSource: cannot open QIODevice data source"); return false; } connect(m_device, SIGNAL(aboutToClose()), SLOT(sourceDataClosed())); connect(m_device, SIGNAL(readChannelFinished()), SLOT(sourceDataClosed())); connect(m_device, SIGNAL(destroyed()), SLOT(sourceDataClosed())); return true; } void QNmeaPositionInfoSourcePrivate::sourceDataClosed() { if (m_nmeaReader && m_device && m_device->bytesAvailable()) m_nmeaReader->readAvailableData(); } void QNmeaPositionInfoSourcePrivate::readyRead() { if (m_nmeaReader) m_nmeaReader->readAvailableData(); } bool QNmeaPositionInfoSourcePrivate::initialize() { if (m_nmeaReader) return true; if (!openSourceDevice()) return false; if (m_updateMode == QNmeaPositionInfoSource::RealTimeMode) m_nmeaReader = new QNmeaRealTimeReader(this); else m_nmeaReader = new QNmeaSimulatedReader(this); return true; } void QNmeaPositionInfoSourcePrivate::prepareSourceDevice() { // some data may already be available if (m_updateMode == QNmeaPositionInfoSource::SimulationMode) { if (m_nmeaReader && m_device->bytesAvailable()) m_nmeaReader->readAvailableData(); } if (!m_connectedReadyRead) { connect(m_device, SIGNAL(readyRead()), SLOT(readyRead())); m_connectedReadyRead = true; } } bool QNmeaPositionInfoSourcePrivate::parsePosInfoFromNmeaData(const char *data, int size, QGeoPositionInfo *posInfo, bool *hasFix) { return m_source->parsePosInfoFromNmeaData(data, size, posInfo, hasFix); } void QNmeaPositionInfoSourcePrivate::startUpdates() { if (m_invokedStart) return; m_invokedStart = true; m_pendingUpdate = QGeoPositionInfo(); m_noUpdateLastInterval = false; bool initialized = initialize(); if (!initialized) return; if (m_updateMode == QNmeaPositionInfoSource::RealTimeMode) { // skip over any buffered data - we only want the newest data if (m_device->bytesAvailable()) { if (m_device->isSequential()) m_device->readAll(); else m_device->seek(m_device->bytesAvailable()); } } if (m_updateTimer) m_updateTimer->stop(); if (m_source->updateInterval() > 0) { if (!m_updateTimer) m_updateTimer = new QBasicTimer; m_updateTimer->start(m_source->updateInterval(), this); } if (initialized) prepareSourceDevice(); } void QNmeaPositionInfoSourcePrivate::stopUpdates() { m_invokedStart = false; if (m_updateTimer) m_updateTimer->stop(); m_pendingUpdate = QGeoPositionInfo(); m_noUpdateLastInterval = false; } void QNmeaPositionInfoSourcePrivate::requestUpdate(int msec) { if (m_requestTimer && m_requestTimer->isActive()) return; if (msec <= 0 || msec < m_source->minimumUpdateInterval()) { emit m_source->updateTimeout(); return; } if (!m_requestTimer) { m_requestTimer = new QTimer(this); connect(m_requestTimer, SIGNAL(timeout()), SLOT(updateRequestTimeout())); } bool initialized = initialize(); if (!initialized) { emit m_source->updateTimeout(); return; } m_requestTimer->start(msec); if (initialized) prepareSourceDevice(); } void QNmeaPositionInfoSourcePrivate::updateRequestTimeout() { m_requestTimer->stop(); emit m_source->updateTimeout(); } void QNmeaPositionInfoSourcePrivate::notifyNewUpdate(QGeoPositionInfo *update, bool hasFix) { // include before uncommenting //qDebug() << "QNmeaPositionInfoSourcePrivate::notifyNewUpdate()" << update->timestamp() << hasFix << m_invokedStart << (m_requestTimer && m_requestTimer->isActive()); QDate date = update->timestamp().date(); if (date.isValid()) { m_currentDate = date; } else { // some sentence have time but no date QTime time = update->timestamp().time(); if (time.isValid() && m_currentDate.isValid()) update->setTimestamp(QDateTime(m_currentDate, time, Qt::UTC)); } // Some attributes are sent in separate NMEA sentences. Save and restore the accuracy // measurements. if (update->hasAttribute(QGeoPositionInfo::HorizontalAccuracy)) m_horizontalAccuracy = update->attribute(QGeoPositionInfo::HorizontalAccuracy); else if (!qIsNaN(m_horizontalAccuracy)) update->setAttribute(QGeoPositionInfo::HorizontalAccuracy, m_horizontalAccuracy); if (update->hasAttribute(QGeoPositionInfo::VerticalAccuracy)) m_verticalAccuracy = update->attribute(QGeoPositionInfo::VerticalAccuracy); else if (!qIsNaN(m_verticalAccuracy)) update->setAttribute(QGeoPositionInfo::VerticalAccuracy, m_verticalAccuracy); if (hasFix && update->isValid()) { if (m_requestTimer && m_requestTimer->isActive()) { m_requestTimer->stop(); emitUpdated(*update); } else if (m_invokedStart) { if (m_updateTimer && m_updateTimer->isActive()) { // for periodic updates, only want the most recent update m_pendingUpdate = *update; if (m_noUpdateLastInterval) { emitPendingUpdate(); m_noUpdateLastInterval = false; } } else { emitUpdated(*update); } } m_lastUpdate = *update; } } void QNmeaPositionInfoSourcePrivate::timerEvent(QTimerEvent *) { emitPendingUpdate(); } void QNmeaPositionInfoSourcePrivate::emitPendingUpdate() { if (m_pendingUpdate.isValid()) { m_updateTimeoutSent = false; m_noUpdateLastInterval = false; emitUpdated(m_pendingUpdate); m_pendingUpdate = QGeoPositionInfo(); } else { if (m_noUpdateLastInterval && !m_updateTimeoutSent) { m_updateTimeoutSent = true; m_pendingUpdate = QGeoPositionInfo(); emit m_source->updateTimeout(); } m_noUpdateLastInterval = true; } } void QNmeaPositionInfoSourcePrivate::emitUpdated(const QGeoPositionInfo &update) { m_lastUpdate = update; emit m_source->positionUpdated(update); } //========================================================= /*! \class QNmeaPositionInfoSource \inmodule QtPositioning \ingroup QtPositioning-positioning \since 5.2 \brief The QNmeaPositionInfoSource class provides positional information using a NMEA data source. NMEA is a commonly used protocol for the specification of one's global position at a certain point in time. The QNmeaPositionInfoSource class reads NMEA data and uses it to provide positional data in the form of QGeoPositionInfo objects. A QNmeaPositionInfoSource instance operates in either \l {RealTimeMode} or \l {SimulationMode}. These modes allow NMEA data to be read from either a live source of positional data, or replayed for simulation purposes from previously recorded NMEA data. The source of NMEA data is set with setDevice(). Use startUpdates() to start receiving regular position updates and stopUpdates() to stop these updates. If you only require updates occasionally, you can call requestUpdate() to request a single update. In both cases the position information is received via the positionUpdated() signal and the last known position can be accessed with lastKnownPosition(). QNmeaPositionInfoSource supports reporting the accuracy of the horizontal and vertical position. To enable position accuracy reporting an estimate of the User Equivalent Range Error associated with the NMEA source must be set with setUserEquivalentRangeError(). */ /*! \enum QNmeaPositionInfoSource::UpdateMode Defines the available update modes. \value RealTimeMode Positional data is read and distributed from the data source as it becomes available. Use this mode if you are using a live source of positional data (for example, a GPS hardware device). \value SimulationMode The data and time information in the NMEA source data is used to provide positional updates at the rate at which the data was originally recorded. Use this mode if the data source contains previously recorded NMEA data and you want to replay the data for simulation purposes. */ /*! Constructs a QNmeaPositionInfoSource instance with the given \a parent and \a updateMode. */ QNmeaPositionInfoSource::QNmeaPositionInfoSource(UpdateMode updateMode, QObject *parent) : QGeoPositionInfoSource(parent), d(new QNmeaPositionInfoSourcePrivate(this, updateMode)) { } /*! Destroys the position source. */ QNmeaPositionInfoSource::~QNmeaPositionInfoSource() { delete d; } /*! Sets the User Equivalent Range Error (UERE) to \a uere. The UERE is used in calculating an estimate of the accuracy of the position information reported by the position info source. The UERE should be set to a value appropriate for the GPS device which generated the NMEA stream. The true UERE value is calculated from multiple error sources including errors introduced by the satellites and signal propogation delays through the atmosphere as well as errors introduced by the receiving GPS equipment. For details on GPS accuracy see \l {http://edu-observatory.org/gps/gps_accuracy.html}. A typical value for UERE is approximately 5.1. \since 5.3 \sa userEquivalentRangeError() */ void QNmeaPositionInfoSource::setUserEquivalentRangeError(double uere) { d->m_userEquivalentRangeError = uere; } /*! Returns the current User Equivalent Range Error (UERE). The UERE is used in calculating an estimate of the accuracy of the position information reported by the position info source. The default value is NaN which means no accuracy information will be provided. \since 5.3 \sa setUserEquivalentRangeError() */ double QNmeaPositionInfoSource::userEquivalentRangeError() const { return d->m_userEquivalentRangeError; } /*! Parses an NMEA sentence string into a QGeoPositionInfo. The default implementation will parse standard NMEA sentences. This method should be reimplemented in a subclass whenever the need to deal with non-standard NMEA sentences arises. The parser reads \a size bytes from \a data and uses that information to setup \a posInfo and \a hasFix. If \a hasFix is set to false then \a posInfo may contain only the time or the date and the time. Returns true if the sentence was succsesfully parsed, otherwise returns false and should not modifiy \a posInfo or \a hasFix. */ bool QNmeaPositionInfoSource::parsePosInfoFromNmeaData(const char *data, int size, QGeoPositionInfo *posInfo, bool *hasFix) { return QLocationUtils::getPosInfoFromNmea(data, size, posInfo, d->m_userEquivalentRangeError, hasFix); } /*! Returns the update mode. */ QNmeaPositionInfoSource::UpdateMode QNmeaPositionInfoSource::updateMode() const { return d->m_updateMode; } /*! Sets the NMEA data source to \a device. If the device is not open, it will be opened in QIODevice::ReadOnly mode. The source device can only be set once and must be set before calling startUpdates() or requestUpdate(). \b {Note:} The \a device must emit QIODevice::readyRead() for the source to be notified when data is available for reading. QNmeaPositionInfoSource does not assume the ownership of the device, and hence does not deallocate it upon destruction. */ void QNmeaPositionInfoSource::setDevice(QIODevice *device) { if (device != d->m_device) { if (!d->m_device) d->m_device = device; else qWarning("QNmeaPositionInfoSource: source device has already been set"); } } /*! Returns the NMEA data source. */ QIODevice *QNmeaPositionInfoSource::device() const { return d->m_device; } /*! \reimp */ void QNmeaPositionInfoSource::setUpdateInterval(int msec) { int interval = msec; if (interval != 0) interval = qMax(msec, minimumUpdateInterval()); QGeoPositionInfoSource::setUpdateInterval(interval); if (d->m_invokedStart) { d->stopUpdates(); d->startUpdates(); } } /*! \reimp */ void QNmeaPositionInfoSource::startUpdates() { d->startUpdates(); } /*! \reimp */ void QNmeaPositionInfoSource::stopUpdates() { d->stopUpdates(); } /*! \reimp */ void QNmeaPositionInfoSource::requestUpdate(int msec) { d->requestUpdate(msec == 0 ? 60000 * 5 : msec); } /*! \reimp */ QGeoPositionInfo QNmeaPositionInfoSource::lastKnownPosition(bool) const { // the bool value does not matter since we only use satellite positioning return d->m_lastUpdate; } /*! \reimp */ QGeoPositionInfoSource::PositioningMethods QNmeaPositionInfoSource::supportedPositioningMethods() const { return SatellitePositioningMethods; } /*! \reimp */ int QNmeaPositionInfoSource::minimumUpdateInterval() const { return 100; } /*! \reimp */ QGeoPositionInfoSource::Error QNmeaPositionInfoSource::error() const { return d->m_positionError; } void QNmeaPositionInfoSource::setError(QGeoPositionInfoSource::Error positionError) { d->m_positionError = positionError; emit QGeoPositionInfoSource::error(positionError); } QT_END_NAMESPACE