Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
Q
qgroundcontrol
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Valentin Platzgummer
qgroundcontrol
Commits
158bfc91
Commit
158bfc91
authored
Aug 28, 2014
by
Lorenz Meier
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #849 from DonLakeFlyer/FTP
Modified to use new FILE_TRANSFER_PROTOCOL message
parents
3e0ca9f6
85e81a84
Changes
6
Hide whitespace changes
Inline
Side-by-side
Showing
6 changed files
with
119 additions
and
83 deletions
+119
-83
MockMavlinkFileServer.cc
src/qgcunittest/MockMavlinkFileServer.cc
+21
-17
MockMavlinkFileServer.h
src/qgcunittest/MockMavlinkFileServer.h
+6
-1
QGCUASFileManagerTest.cc
src/qgcunittest/QGCUASFileManagerTest.cc
+5
-6
QGCUASFileManagerTest.h
src/qgcunittest/QGCUASFileManagerTest.h
+4
-0
QGCUASFileManager.cc
src/uas/QGCUASFileManager.cc
+45
-27
QGCUASFileManager.h
src/uas/QGCUASFileManager.h
+38
-32
No files found.
src/qgcunittest/MockMavlinkFileServer.cc
View file @
158bfc91
...
@@ -45,8 +45,10 @@ const MockMavlinkFileServer::FileTestCase MockMavlinkFileServer::rgFileTestCases
...
@@ -45,8 +45,10 @@ const MockMavlinkFileServer::FileTestCase MockMavlinkFileServer::rgFileTestCases
// We only support a single fixed session
// We only support a single fixed session
const
uint8_t
MockMavlinkFileServer
::
_sessionId
=
1
;
const
uint8_t
MockMavlinkFileServer
::
_sessionId
=
1
;
MockMavlinkFileServer
::
MockMavlinkFileServer
(
void
)
:
MockMavlinkFileServer
::
MockMavlinkFileServer
(
uint8_t
systemIdQGC
,
uint8_t
systemIdServer
)
:
_errMode
(
errModeNone
)
_errMode
(
errModeNone
),
_systemIdServer
(
systemIdServer
),
_systemIdQGC
(
systemIdQGC
)
{
{
}
}
...
@@ -74,7 +76,6 @@ void MockMavlinkFileServer::_listCommand(QGCUASFileManager::Request* request, ui
...
@@ -74,7 +76,6 @@ void MockMavlinkFileServer::_listCommand(QGCUASFileManager::Request* request, ui
return
;
return
;
}
}
ackResponse
.
hdr
.
magic
=
'f'
;
ackResponse
.
hdr
.
opcode
=
QGCUASFileManager
::
kRspAck
;
ackResponse
.
hdr
.
opcode
=
QGCUASFileManager
::
kRspAck
;
ackResponse
.
hdr
.
session
=
0
;
ackResponse
.
hdr
.
session
=
0
;
ackResponse
.
hdr
.
offset
=
request
->
hdr
.
offset
;
ackResponse
.
hdr
.
offset
=
request
->
hdr
.
offset
;
...
@@ -133,7 +134,6 @@ void MockMavlinkFileServer::_openCommand(QGCUASFileManager::Request* request, ui
...
@@ -133,7 +134,6 @@ void MockMavlinkFileServer::_openCommand(QGCUASFileManager::Request* request, ui
return
;
return
;
}
}
response
.
hdr
.
magic
=
'f'
;
response
.
hdr
.
opcode
=
QGCUASFileManager
::
kRspAck
;
response
.
hdr
.
opcode
=
QGCUASFileManager
::
kRspAck
;
response
.
hdr
.
session
=
_sessionId
;
response
.
hdr
.
session
=
_sessionId
;
...
@@ -183,7 +183,6 @@ void MockMavlinkFileServer::_readCommand(QGCUASFileManager::Request* request, ui
...
@@ -183,7 +183,6 @@ void MockMavlinkFileServer::_readCommand(QGCUASFileManager::Request* request, ui
// We should always have written something, otherwise there is something wrong with the code above
// We should always have written something, otherwise there is something wrong with the code above
Q_ASSERT
(
cDataBytes
);
Q_ASSERT
(
cDataBytes
);
response
.
hdr
.
magic
=
'f'
;
response
.
hdr
.
session
=
_sessionId
;
response
.
hdr
.
session
=
_sessionId
;
response
.
hdr
.
size
=
cDataBytes
;
response
.
hdr
.
size
=
cDataBytes
;
response
.
hdr
.
offset
=
request
->
hdr
.
offset
;
response
.
hdr
.
offset
=
request
->
hdr
.
offset
;
...
@@ -214,15 +213,15 @@ void MockMavlinkFileServer::sendMessage(mavlink_message_t message)
...
@@ -214,15 +213,15 @@ void MockMavlinkFileServer::sendMessage(mavlink_message_t message)
{
{
QGCUASFileManager
::
Request
ackResponse
;
QGCUASFileManager
::
Request
ackResponse
;
Q_ASSERT
(
message
.
msgid
==
MAVLINK_MSG_ID_
ENCAPSULATED_DATA
);
Q_ASSERT
(
message
.
msgid
==
MAVLINK_MSG_ID_
FILE_TRANSFER_PROTOCOL
);
mavlink_
encapsulated_data_t
requestEncapsulatedData
;
mavlink_
file_transfer_protocol_t
requestFileTransferProtocol
;
mavlink_msg_
encapsulated_data_decode
(
&
message
,
&
requestEncapsulatedData
);
mavlink_msg_
file_transfer_protocol_decode
(
&
message
,
&
requestFileTransferProtocol
);
QGCUASFileManager
::
Request
*
request
=
(
QGCUASFileManager
::
Request
*
)
&
request
EncapsulatedData
.
data
[
0
];
QGCUASFileManager
::
Request
*
request
=
(
QGCUASFileManager
::
Request
*
)
&
request
FileTransferProtocol
.
payload
[
0
];
Q_ASSERT
(
request
->
hdr
.
magic
==
QGCUASFileManager
::
kProtocolMagic
);
Q_ASSERT
(
request
FileTransferProtocol
.
target_system
==
_systemIdServer
);
uint16_t
incomingSeqNumber
=
request
EncapsulatedData
.
seqn
r
;
uint16_t
incomingSeqNumber
=
request
->
hdr
.
seqNumbe
r
;
uint16_t
outgoingSeqNumber
=
_nextSeqNumber
(
incomingSeqNumber
);
uint16_t
outgoingSeqNumber
=
_nextSeqNumber
(
incomingSeqNumber
);
if
(
_errMode
==
errModeNoResponse
)
{
if
(
_errMode
==
errModeNoResponse
)
{
...
@@ -236,6 +235,7 @@ void MockMavlinkFileServer::sendMessage(mavlink_message_t message)
...
@@ -236,6 +235,7 @@ void MockMavlinkFileServer::sendMessage(mavlink_message_t message)
// Validate CRC
// Validate CRC
if
(
request
->
hdr
.
crc32
!=
QGCUASFileManager
::
crc32
(
request
))
{
if
(
request
->
hdr
.
crc32
!=
QGCUASFileManager
::
crc32
(
request
))
{
qDebug
()
<<
"Bad CRC received - opcode:"
<<
request
->
hdr
.
opcode
<<
"expected:"
<<
request
->
hdr
.
crc32
<<
"actual:"
<<
QGCUASFileManager
::
crc32
(
request
);
_sendNak
(
QGCUASFileManager
::
kErrCrc
,
outgoingSeqNumber
);
_sendNak
(
QGCUASFileManager
::
kErrCrc
,
outgoingSeqNumber
);
}
}
...
@@ -250,7 +250,6 @@ void MockMavlinkFileServer::sendMessage(mavlink_message_t message)
...
@@ -250,7 +250,6 @@ void MockMavlinkFileServer::sendMessage(mavlink_message_t message)
case
QGCUASFileManager
:
:
kCmdNone
:
case
QGCUASFileManager
:
:
kCmdNone
:
// ignored, always acked
// ignored, always acked
ackResponse
.
hdr
.
magic
=
'f'
;
ackResponse
.
hdr
.
opcode
=
QGCUASFileManager
::
kRspAck
;
ackResponse
.
hdr
.
opcode
=
QGCUASFileManager
::
kRspAck
;
ackResponse
.
hdr
.
session
=
0
;
ackResponse
.
hdr
.
session
=
0
;
ackResponse
.
hdr
.
crc32
=
0
;
ackResponse
.
hdr
.
crc32
=
0
;
...
@@ -294,7 +293,6 @@ void MockMavlinkFileServer::_sendAck(uint16_t seqNumber)
...
@@ -294,7 +293,6 @@ void MockMavlinkFileServer::_sendAck(uint16_t seqNumber)
{
{
QGCUASFileManager
::
Request
ackResponse
;
QGCUASFileManager
::
Request
ackResponse
;
ackResponse
.
hdr
.
magic
=
'f'
;
ackResponse
.
hdr
.
opcode
=
QGCUASFileManager
::
kRspAck
;
ackResponse
.
hdr
.
opcode
=
QGCUASFileManager
::
kRspAck
;
ackResponse
.
hdr
.
session
=
0
;
ackResponse
.
hdr
.
session
=
0
;
ackResponse
.
hdr
.
size
=
0
;
ackResponse
.
hdr
.
size
=
0
;
...
@@ -307,7 +305,6 @@ void MockMavlinkFileServer::_sendNak(QGCUASFileManager::ErrorCode error, uint16_
...
@@ -307,7 +305,6 @@ void MockMavlinkFileServer::_sendNak(QGCUASFileManager::ErrorCode error, uint16_
{
{
QGCUASFileManager
::
Request
nakResponse
;
QGCUASFileManager
::
Request
nakResponse
;
nakResponse
.
hdr
.
magic
=
'f'
;
nakResponse
.
hdr
.
opcode
=
QGCUASFileManager
::
kRspNak
;
nakResponse
.
hdr
.
opcode
=
QGCUASFileManager
::
kRspNak
;
nakResponse
.
hdr
.
session
=
0
;
nakResponse
.
hdr
.
session
=
0
;
nakResponse
.
hdr
.
size
=
1
;
nakResponse
.
hdr
.
size
=
1
;
...
@@ -321,14 +318,21 @@ void MockMavlinkFileServer::_emitResponse(QGCUASFileManager::Request* request, u
...
@@ -321,14 +318,21 @@ void MockMavlinkFileServer::_emitResponse(QGCUASFileManager::Request* request, u
{
{
mavlink_message_t
mavlinkMessage
;
mavlink_message_t
mavlinkMessage
;
request
->
hdr
.
magic
=
QGCUASFileManager
::
kProtocolMagic
;
request
->
hdr
.
seqNumber
=
seqNumber
;
request
->
hdr
.
crc32
=
QGCUASFileManager
::
crc32
(
request
);
request
->
hdr
.
crc32
=
QGCUASFileManager
::
crc32
(
request
);
if
(
_errMode
==
errModeBadCRC
)
{
if
(
_errMode
==
errModeBadCRC
)
{
// Return a bad CRC
// Return a bad CRC
request
->
hdr
.
crc32
++
;
request
->
hdr
.
crc32
++
;
}
}
mavlink_msg_encapsulated_data_pack
(
250
,
50
,
&
mavlinkMessage
,
seqNumber
,
(
uint8_t
*
)
request
);
mavlink_msg_file_transfer_protocol_pack
(
_systemIdServer
,
// System ID
0
,
// Component ID
&
mavlinkMessage
,
// Mavlink Message to pack into
0
,
// Target network
_systemIdQGC
,
// QGC Target System ID
0
,
// Target component
(
uint8_t
*
)
request
);
// Payload
emit
messageReceived
(
NULL
,
mavlinkMessage
);
emit
messageReceived
(
NULL
,
mavlinkMessage
);
}
}
...
...
src/qgcunittest/MockMavlinkFileServer.h
View file @
158bfc91
...
@@ -40,7 +40,10 @@ class MockMavlinkFileServer : public MockMavlinkInterface
...
@@ -40,7 +40,10 @@ class MockMavlinkFileServer : public MockMavlinkInterface
Q_OBJECT
Q_OBJECT
public:
public:
MockMavlinkFileServer
(
void
);
/// @brief Constructor for MockMavlinkFileServer
/// @param System ID for QGroundControl App
/// @pqram System ID for this Server
MockMavlinkFileServer
(
uint8_t
systemIdQGC
,
uint8_t
systemIdServer
);
/// @brief Sets the list of files returned by the List command. Prepend names with F or D
/// @brief Sets the list of files returned by the List command. Prepend names with F or D
/// to indicate (F)ile or (D)irectory.
/// to indicate (F)ile or (D)irectory.
...
@@ -103,6 +106,8 @@ private:
...
@@ -103,6 +106,8 @@ private:
static
const
uint8_t
_sessionId
;
static
const
uint8_t
_sessionId
;
uint8_t
_readFileLength
;
///< Length of active file being read
uint8_t
_readFileLength
;
///< Length of active file being read
ErrorMode_t
_errMode
;
///< Currently set error mode, as specified by setErrorMode
ErrorMode_t
_errMode
;
///< Currently set error mode, as specified by setErrorMode
const
uint8_t
_systemIdServer
;
///< System ID for server
const
uint8_t
_systemIdQGC
;
///< QGC System ID
};
};
#endif
#endif
src/qgcunittest/QGCUASFileManagerTest.cc
View file @
158bfc91
...
@@ -31,6 +31,7 @@
...
@@ -31,6 +31,7 @@
/// @author Don Gagne <don@thegagnes.com>
/// @author Don Gagne <don@thegagnes.com>
QGCUASFileManagerUnitTest
::
QGCUASFileManagerUnitTest
(
void
)
:
QGCUASFileManagerUnitTest
::
QGCUASFileManagerUnitTest
(
void
)
:
_mockFileServer
(
_systemIdQGC
,
_systemIdServer
),
_fileManager
(
NULL
),
_fileManager
(
NULL
),
_multiSpy
(
NULL
)
_multiSpy
(
NULL
)
{
{
...
@@ -40,6 +41,7 @@ QGCUASFileManagerUnitTest::QGCUASFileManagerUnitTest(void) :
...
@@ -40,6 +41,7 @@ QGCUASFileManagerUnitTest::QGCUASFileManagerUnitTest(void) :
// Called once before all test cases are run
// Called once before all test cases are run
void
QGCUASFileManagerUnitTest
::
initTestCase
(
void
)
void
QGCUASFileManagerUnitTest
::
initTestCase
(
void
)
{
{
_mockUAS
.
setMockSystemId
(
_systemIdServer
);
_mockUAS
.
setMockMavlinkPlugin
(
&
_mockFileServer
);
_mockUAS
.
setMockMavlinkPlugin
(
&
_mockFileServer
);
}
}
...
@@ -48,19 +50,16 @@ void QGCUASFileManagerUnitTest::init(void)
...
@@ -48,19 +50,16 @@ void QGCUASFileManagerUnitTest::init(void)
{
{
Q_ASSERT
(
_multiSpy
==
NULL
);
Q_ASSERT
(
_multiSpy
==
NULL
);
_fileManager
=
new
QGCUASFileManager
(
NULL
,
&
_mockUAS
);
_fileManager
=
new
QGCUASFileManager
(
NULL
,
&
_mockUAS
,
_systemIdQGC
);
Q_CHECK_PTR
(
_fileManager
);
Q_CHECK_PTR
(
_fileManager
);
// Reset any internal state back to normal
// Reset any internal state back to normal
_mockFileServer
.
setErrorMode
(
MockMavlinkFileServer
::
errModeNone
);
_mockFileServer
.
setErrorMode
(
MockMavlinkFileServer
::
errModeNone
);
_fileListReceived
.
clear
();
_fileListReceived
.
clear
();
bool
connected
=
connect
(
&
_mockFileServer
,
SIGNAL
(
messageReceived
(
LinkInterface
*
,
mavlink_message_t
)),
_fileManager
,
SLOT
(
receiveMessage
(
LinkInterface
*
,
mavlink_message_t
)));
connect
(
&
_mockFileServer
,
&
MockMavlinkFileServer
::
messageReceived
,
_fileManager
,
&
QGCUASFileManager
::
receiveMessage
);
Q_ASSERT
(
connected
);
Q_UNUSED
(
connected
);
// Silent release build compiler warning
connected
=
connect
(
_fileManager
,
SIGNAL
(
listEntry
(
const
QString
&
)),
this
,
SLOT
(
listEntry
(
const
QString
&
)));
connect
(
_fileManager
,
&
QGCUASFileManager
::
listEntry
,
this
,
&
QGCUASFileManagerUnitTest
::
listEntry
);
Q_ASSERT
(
connected
);
_rgSignals
[
listEntrySignalIndex
]
=
SIGNAL
(
listEntry
(
const
QString
&
));
_rgSignals
[
listEntrySignalIndex
]
=
SIGNAL
(
listEntry
(
const
QString
&
));
_rgSignals
[
listCompleteSignalIndex
]
=
SIGNAL
(
listComplete
(
void
));
_rgSignals
[
listCompleteSignalIndex
]
=
SIGNAL
(
listComplete
(
void
));
...
...
src/qgcunittest/QGCUASFileManagerTest.h
View file @
158bfc91
...
@@ -80,6 +80,10 @@ private:
...
@@ -80,6 +80,10 @@ private:
downloadFileCompleteSignalMask
=
1
<<
downloadFileCompleteSignalIndex
,
downloadFileCompleteSignalMask
=
1
<<
downloadFileCompleteSignalIndex
,
errorMessageSignalMask
=
1
<<
errorMessageSignalIndex
,
errorMessageSignalMask
=
1
<<
errorMessageSignalIndex
,
};
};
static
const
uint8_t
_systemIdQGC
=
255
;
static
const
uint8_t
_systemIdServer
=
128
;
MockUAS
_mockUAS
;
MockUAS
_mockUAS
;
MockMavlinkFileServer
_mockFileServer
;
MockMavlinkFileServer
_mockFileServer
;
...
...
src/uas/QGCUASFileManager.cc
View file @
158bfc91
...
@@ -24,6 +24,7 @@
...
@@ -24,6 +24,7 @@
#include "QGCUASFileManager.h"
#include "QGCUASFileManager.h"
#include "QGC.h"
#include "QGC.h"
#include "MAVLinkProtocol.h"
#include "MAVLinkProtocol.h"
#include "MainWindow.h"
#include <QFile>
#include <QFile>
#include <QDir>
#include <QDir>
...
@@ -66,16 +67,20 @@ static const quint32 crctab[] =
...
@@ -66,16 +67,20 @@ static const quint32 crctab[] =
};
};
QGCUASFileManager
::
QGCUASFileManager
(
QObject
*
parent
,
UASInterface
*
uas
)
:
QGCUASFileManager
::
QGCUASFileManager
(
QObject
*
parent
,
UASInterface
*
uas
,
uint8_t
unitTestSystemIdQGC
)
:
QObject
(
parent
),
QObject
(
parent
),
_currentOperation
(
kCOIdle
),
_currentOperation
(
kCOIdle
),
_mav
(
uas
),
_mav
(
uas
),
_lastOutgoingSeqNumber
(
0
),
_lastOutgoingSeqNumber
(
0
),
_activeSession
(
0
)
_activeSession
(
0
),
_systemIdQGC
(
unitTestSystemIdQGC
)
{
{
bool
connected
=
connect
(
&
_ackTimer
,
SIGNAL
(
timeout
()),
this
,
SLOT
(
_ackTimeout
()));
connect
(
&
_ackTimer
,
&
QTimer
::
timeout
,
this
,
&
QGCUASFileManager
::
_ackTimeout
);
Q_ASSERT
(
connected
);
Q_UNUSED
(
connected
);
// Silence retail unused variable error
_systemIdServer
=
_mav
->
getUASID
();
// Make sure we don't have bad structure packing
Q_ASSERT
(
sizeof
(
RequestHeader
)
==
16
);
}
}
/// @brief Calculates a 32 bit CRC for the specified request.
/// @brief Calculates a 32 bit CRC for the specified request.
...
@@ -90,6 +95,9 @@ quint32 QGCUASFileManager::crc32(Request* request, unsigned state)
...
@@ -90,6 +95,9 @@ quint32 QGCUASFileManager::crc32(Request* request, unsigned state)
// Always calculate CRC with 0 initial CRC value
// Always calculate CRC with 0 initial CRC value
quint32
crcSave
=
request
->
hdr
.
crc32
;
quint32
crcSave
=
request
->
hdr
.
crc32
;
request
->
hdr
.
crc32
=
0
;
request
->
hdr
.
crc32
=
0
;
request
->
hdr
.
padding
[
0
]
=
0
;
request
->
hdr
.
padding
[
1
]
=
0
;
request
->
hdr
.
padding
[
2
]
=
0
;
for
(
size_t
i
=
0
;
i
<
cbData
;
i
++
)
for
(
size_t
i
=
0
;
i
<
cbData
;
i
++
)
state
=
crctab
[(
state
^
data
[
i
])
&
0xff
]
^
(
state
>>
8
);
state
=
crctab
[(
state
^
data
[
i
])
&
0xff
]
^
(
state
>>
8
);
...
@@ -182,6 +190,7 @@ void QGCUASFileManager::_readAckResponse(Request* readAck)
...
@@ -182,6 +190,7 @@ void QGCUASFileManager::_readAckResponse(Request* readAck)
request
.
hdr
.
session
=
_activeSession
;
request
.
hdr
.
session
=
_activeSession
;
request
.
hdr
.
opcode
=
kCmdRead
;
request
.
hdr
.
opcode
=
kCmdRead
;
request
.
hdr
.
offset
=
_readOffset
;
request
.
hdr
.
offset
=
_readOffset
;
request
.
hdr
.
size
=
0
;
_sendRequest
(
&
request
);
_sendRequest
(
&
request
);
}
else
{
}
else
{
...
@@ -252,28 +261,25 @@ void QGCUASFileManager::receiveMessage(LinkInterface* link, mavlink_message_t me
...
@@ -252,28 +261,25 @@ void QGCUASFileManager::receiveMessage(LinkInterface* link, mavlink_message_t me
{
{
Q_UNUSED
(
link
);
Q_UNUSED
(
link
);
if
(
message
.
msgid
!=
MAVLINK_MSG_ID_ENCAPSULATED_DATA
)
{
// receiveMessage is signalled will all mavlink messages so we need to filter everything else out but ours.
// wtf, not for us
if
(
message
.
msgid
!=
MAVLINK_MSG_ID_FILE_TRANSFER_PROTOCOL
)
{
return
;
}
// XXX: hack to prevent files from videostream to interfere
// FIXME: magic number
if
(
message
.
compid
!=
50
)
{
return
;
return
;
}
}
mavlink_encapsulated_data_t
data
;
mavlink_msg_encapsulated_data_decode
(
&
message
,
&
data
);
Request
*
request
=
(
Request
*
)
&
data
.
data
[
0
];
if
(
request
->
hdr
.
magic
!=
kProtocolMagic
)
{
mavlink_file_transfer_protocol_t
data
;
mavlink_msg_file_transfer_protocol_decode
(
&
message
,
&
data
);
// Make sure we are the target system
if
(
data
.
target_system
!=
_systemIdQGC
)
{
qDebug
()
<<
"Received MAVLINK_MSG_ID_FILE_TRANSFER_PROTOCOL with possibly incorrect system id"
<<
_systemIdQGC
;
return
;
return
;
}
}
Request
*
request
=
(
Request
*
)
&
data
.
payload
[
0
];
_clearAckTimeout
();
_clearAckTimeout
();
uint16_t
incomingSeqNumber
=
data
.
seqn
r
;
uint16_t
incomingSeqNumber
=
request
->
hdr
.
seqNumbe
r
;
// Make sure we have a good CRC
// Make sure we have a good CRC
quint32
expectedCRC
=
crc32
(
request
);
quint32
expectedCRC
=
crc32
(
request
);
...
@@ -381,10 +387,10 @@ void QGCUASFileManager::_sendListCommand(void)
...
@@ -381,10 +387,10 @@ void QGCUASFileManager::_sendListCommand(void)
{
{
Request
request
;
Request
request
;
request
.
hdr
.
magic
=
'f'
;
request
.
hdr
.
session
=
0
;
request
.
hdr
.
session
=
0
;
request
.
hdr
.
opcode
=
kCmdList
;
request
.
hdr
.
opcode
=
kCmdList
;
request
.
hdr
.
offset
=
_listOffset
;
request
.
hdr
.
offset
=
_listOffset
;
request
.
hdr
.
size
=
0
;
_fillRequestWithString
(
&
request
,
_listPath
);
_fillRequestWithString
(
&
request
,
_listPath
);
...
@@ -416,10 +422,10 @@ void QGCUASFileManager::downloadPath(const QString& from, const QDir& downloadDi
...
@@ -416,10 +422,10 @@ void QGCUASFileManager::downloadPath(const QString& from, const QDir& downloadDi
_currentOperation
=
kCOOpen
;
_currentOperation
=
kCOOpen
;
Request
request
;
Request
request
;
request
.
hdr
.
magic
=
'f'
;
request
.
hdr
.
session
=
0
;
request
.
hdr
.
session
=
0
;
request
.
hdr
.
opcode
=
kCmdOpen
;
request
.
hdr
.
opcode
=
kCmdOpen
;
request
.
hdr
.
offset
=
0
;
request
.
hdr
.
offset
=
0
;
request
.
hdr
.
size
=
0
;
_fillRequestWithString
(
&
request
,
from
);
_fillRequestWithString
(
&
request
,
from
);
_sendRequest
(
&
request
);
_sendRequest
(
&
request
);
}
}
...
@@ -470,7 +476,6 @@ bool QGCUASFileManager::_sendOpcodeOnlyCmd(uint8_t opcode, OperationState newOpS
...
@@ -470,7 +476,6 @@ bool QGCUASFileManager::_sendOpcodeOnlyCmd(uint8_t opcode, OperationState newOpS
}
}
Request
request
;
Request
request
;
request
.
hdr
.
magic
=
'f'
;
request
.
hdr
.
session
=
0
;
request
.
hdr
.
session
=
0
;
request
.
hdr
.
opcode
=
opcode
;
request
.
hdr
.
opcode
=
opcode
;
request
.
hdr
.
offset
=
0
;
request
.
hdr
.
offset
=
0
;
...
@@ -523,6 +528,7 @@ void QGCUASFileManager::_sendTerminateCommand(void)
...
@@ -523,6 +528,7 @@ void QGCUASFileManager::_sendTerminateCommand(void)
Request
request
;
Request
request
;
request
.
hdr
.
session
=
_activeSession
;
request
.
hdr
.
session
=
_activeSession
;
request
.
hdr
.
opcode
=
kCmdTerminate
;
request
.
hdr
.
opcode
=
kCmdTerminate
;
request
.
hdr
.
size
=
0
;
_sendRequest
(
&
request
);
_sendRequest
(
&
request
);
}
}
...
@@ -547,10 +553,22 @@ void QGCUASFileManager::_sendRequest(Request* request)
...
@@ -547,10 +553,22 @@ void QGCUASFileManager::_sendRequest(Request* request)
_lastOutgoingSeqNumber
++
;
_lastOutgoingSeqNumber
++
;
request
->
hdr
.
magic
=
kProtocolMagic
;
request
->
hdr
.
seqNumber
=
_lastOutgoingSeqNumber
;
request
->
hdr
.
crc32
=
crc32
(
request
);
request
->
hdr
.
crc32
=
crc32
(
request
);
// FIXME: Send correct system id instead of harcoded 250
// FIXME: What about the component id? Should it be set to something specific.
if
(
_systemIdQGC
==
0
)
{
mavlink_msg_encapsulated_data_pack
(
250
,
0
,
&
message
,
_lastOutgoingSeqNumber
,
(
uint8_t
*
)
request
);
_systemIdQGC
=
MainWindow
::
instance
()
->
getMAVLink
()
->
getSystemId
();
}
Q_ASSERT
(
_mav
);
mavlink_msg_file_transfer_protocol_pack
(
_systemIdQGC
,
// QGC System ID
0
,
// QGC Component ID
&
message
,
// Mavlink Message to pack into
0
,
// Target network
_systemIdServer
,
// Target system
0
,
// Target component
(
uint8_t
*
)
request
);
// Payload
_mav
->
sendMessage
(
message
);
_mav
->
sendMessage
(
message
);
}
}
src/uas/QGCUASFileManager.h
View file @
158bfc91
...
@@ -33,7 +33,7 @@ class QGCUASFileManager : public QObject
...
@@ -33,7 +33,7 @@ class QGCUASFileManager : public QObject
{
{
Q_OBJECT
Q_OBJECT
public:
public:
QGCUASFileManager
(
QObject
*
parent
,
UASInterface
*
uas
);
QGCUASFileManager
(
QObject
*
parent
,
UASInterface
*
uas
,
uint8_t
unitTestSystemIdQGC
=
0
);
/// These methods are only used for testing purposes.
/// These methods are only used for testing purposes.
bool
_sendCmdTestAck
(
void
)
{
return
_sendOpcodeOnlyCmd
(
kCmdNone
,
kCOAck
);
};
bool
_sendCmdTestAck
(
void
)
{
return
_sendOpcodeOnlyCmd
(
kCmdNone
,
kCOAck
);
};
...
@@ -74,26 +74,29 @@ public slots:
...
@@ -74,26 +74,29 @@ public slots:
void
downloadPath
(
const
QString
&
from
,
const
QDir
&
downloadDir
);
void
downloadPath
(
const
QString
&
from
,
const
QDir
&
downloadDir
);
protected:
protected:
static
const
uint8_t
kProtocolMagic
=
'f'
;
/// @brief This is the fixed length portion of the protocol data. Trying to pack structures across differing compilers is
/// questionable, so we pad the structure ourselves to 32 bit alignment which should get us what we want.
struct
RequestHeader
struct
RequestHeader
{
{
uint8_t
magic
;
///> Magic byte 'f' to idenitfy FTP protocol
uint16_t
seqNumber
;
///< sequence number for message
uint8_t
session
;
///> Session id for read and write commands
uint8_t
session
;
///< Session id for read and write commands
uint8_t
opcode
;
///> Command opcode
uint8_t
opcode
;
///< Command opcode
uint8_t
size
;
///> Size of data
uint8_t
size
;
///< Size of data
uint32_t
crc32
;
///> CRC for entire Request structure, with crc32 set to 0
uint8_t
padding
[
3
];
uint32_t
offset
;
///> Offsets for List and Read commands
uint32_t
crc32
;
///< CRC for entire Request structure, with crc32 and padding set to 0
uint32_t
offset
;
///< Offsets for List and Read commands
};
};
struct
Request
struct
Request
{
{
struct
RequestHeader
hdr
;
struct
RequestHeader
hdr
;
// We use a union here instead of just casting (uint32_t)&
data
[0] to not break strict aliasing rules
// We use a union here instead of just casting (uint32_t)&
payload
[0] to not break strict aliasing rules
union
{
union
{
// The entire Request must fit into the
data member of the mavlink_encapsulated_data
_t structure. We use as many leftover bytes
// The entire Request must fit into the
payload member of the mavlink_file_transfer_protocol
_t structure. We use as many leftover bytes
// after we use up space for the RequestHeader for the data portion of the Request.
// after we use up space for the RequestHeader for the data portion of the Request.
uint8_t
data
[
sizeof
(((
mavlink_
encapsulated_data_t
*
)
0
)
->
data
)
-
sizeof
(
RequestHeader
)];
uint8_t
data
[
sizeof
(((
mavlink_
file_transfer_protocol_t
*
)
0
)
->
payload
)
-
sizeof
(
RequestHeader
)];
// File length returned by Open command
// File length returned by Open command
uint32_t
openFileLength
;
uint32_t
openFileLength
;
...
@@ -103,22 +106,22 @@ protected:
...
@@ -103,22 +106,22 @@ protected:
enum
Opcode
enum
Opcode
{
{
// Commands
// Commands
kCmdNone
,
///
>
ignored, always acked
kCmdNone
,
///
<
ignored, always acked
kCmdTerminate
,
///
>
releases sessionID, closes file
kCmdTerminate
,
///
<
releases sessionID, closes file
kCmdReset
,
///
>
terminates all sessions
kCmdReset
,
///
<
terminates all sessions
kCmdList
,
///
>
list files in <path> from <offset>
kCmdList
,
///
<
list files in <path> from <offset>
kCmdOpen
,
///
>
opens <path> for reading, returns <session>
kCmdOpen
,
///
<
opens <path> for reading, returns <session>
kCmdRead
,
///
>
reads <size> bytes from <offset> in <session>
kCmdRead
,
///
<
reads <size> bytes from <offset> in <session>
kCmdCreate
,
///
>
creates <path> for writing, returns <session>
kCmdCreate
,
///
<
creates <path> for writing, returns <session>
kCmdWrite
,
///
>
appends <size> bytes at <offset> in <session>
kCmdWrite
,
///
<
appends <size> bytes at <offset> in <session>
kCmdRemove
,
///
>
remove file (only if created by server?)
kCmdRemove
,
///
<
remove file (only if created by server?)
// Responses
// Responses
kRspAck
,
///
>
positive acknowledgement of previous command
kRspAck
,
///
<
positive acknowledgement of previous command
kRspNak
,
///
>
negative acknowledgement of previous command
kRspNak
,
///
<
negative acknowledgement of previous command
// Used for testing only, not part of protocol
// Used for testing only, not part of protocol
kCmdTestNoAck
,
// ignored, ack not sent back, should timeout waiting for ack
kCmdTestNoAck
,
//
/<
ignored, ack not sent back, should timeout waiting for ack
};
};
enum
ErrorCode
enum
ErrorCode
...
@@ -170,21 +173,24 @@ protected:
...
@@ -170,21 +173,24 @@ protected:
static
quint32
crc32
(
Request
*
request
,
unsigned
state
=
0
);
static
quint32
crc32
(
Request
*
request
,
unsigned
state
=
0
);
static
QString
errorString
(
uint8_t
errorCode
);
static
QString
errorString
(
uint8_t
errorCode
);
OperationState
_currentOperation
;
///
>
Current operation of state machine
OperationState
_currentOperation
;
///
<
Current operation of state machine
QTimer
_ackTimer
;
///
>
Used to signal a timeout waiting for an ack
QTimer
_ackTimer
;
///
<
Used to signal a timeout waiting for an ack
UASInterface
*
_mav
;
UASInterface
*
_mav
;
uint16_t
_lastOutgoingSeqNumber
;
///< Sequence number sent in last outgoing packet
uint16_t
_lastOutgoingSeqNumber
;
///< Sequence number sent in last outgoing packet
unsigned
_listOffset
;
///> offset for the current List operation
unsigned
_listOffset
;
///< offset for the current List operation
QString
_listPath
;
///> path for the current List operation
QString
_listPath
;
///< path for the current List operation
uint8_t
_activeSession
;
///< currently active session, 0 for none
uint32_t
_readOffset
;
///< current read offset
QByteArray
_readFileAccumulator
;
///< Holds file being downloaded
QDir
_readFileDownloadDir
;
///< Directory to download file to
QString
_readFileDownloadFilename
;
///< Filename (no path) for download file
uint8_t
_activeSession
;
///> currently active session, 0 for none
uint8_t
_systemIdQGC
;
///< System ID for QGC
uint32_t
_readOffset
;
///> current read offset
uint8_t
_systemIdServer
;
///< System ID for server
QByteArray
_readFileAccumulator
;
///> Holds file being downloaded
QDir
_readFileDownloadDir
;
///> Directory to download file to
QString
_readFileDownloadFilename
;
///> Filename (no path) for download file
// We give MockMavlinkFileServer friend access so that it can use the data structures and opcodes
// We give MockMavlinkFileServer friend access so that it can use the data structures and opcodes
// to build a mock mavlink file server for testing.
// to build a mock mavlink file server for testing.
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment