QGCUASFileManager.cc 7.8 KB
Newer Older
1 2
#include "QGCUASFileManager.h"
#include "QGC.h"
Lorenz Meier's avatar
Lorenz Meier committed
3 4 5 6
#include "MAVLinkProtocol.h"

#include <QFile>
#include <QDir>
none's avatar
none committed
7
#include <string>
Lorenz Meier's avatar
Lorenz Meier committed
8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43

static const quint32 crctab[] =
{
    0x00000000, 0x77073096, 0xee0e612c, 0x990951ba, 0x076dc419, 0x706af48f, 0xe963a535, 0x9e6495a3,
    0x0edb8832, 0x79dcb8a4, 0xe0d5e91e, 0x97d2d988, 0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, 0x90bf1d91,
    0x1db71064, 0x6ab020f2, 0xf3b97148, 0x84be41de, 0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7,
    0x136c9856, 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec, 0x14015c4f, 0x63066cd9, 0xfa0f3d63, 0x8d080df5,
    0x3b6e20c8, 0x4c69105e, 0xd56041e4, 0xa2677172, 0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b,
    0x35b5a8fa, 0x42b2986c, 0xdbbbc9d6, 0xacbcf940, 0x32d86ce3, 0x45df5c75, 0xdcd60dcf, 0xabd13d59,
    0x26d930ac, 0x51de003a, 0xc8d75180, 0xbfd06116, 0x21b4f4b5, 0x56b3c423, 0xcfba9599, 0xb8bda50f,
    0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924, 0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d,
    0x76dc4190, 0x01db7106, 0x98d220bc, 0xefd5102a, 0x71b18589, 0x06b6b51f, 0x9fbfe4a5, 0xe8b8d433,
    0x7807c9a2, 0x0f00f934, 0x9609a88e, 0xe10e9818, 0x7f6a0dbb, 0x086d3d2d, 0x91646c97, 0xe6635c01,
    0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e, 0x6c0695ed, 0x1b01a57b, 0x8208f4c1, 0xf50fc457,
    0x65b0d9c6, 0x12b7e950, 0x8bbeb8ea, 0xfcb9887c, 0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, 0xfbd44c65,
    0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2, 0x4adfa541, 0x3dd895d7, 0xa4d1c46d, 0xd3d6f4fb,
    0x4369e96a, 0x346ed9fc, 0xad678846, 0xda60b8d0, 0x44042d73, 0x33031de5, 0xaa0a4c5f, 0xdd0d7cc9,
    0x5005713c, 0x270241aa, 0xbe0b1010, 0xc90c2086, 0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f,
    0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4, 0x59b33d17, 0x2eb40d81, 0xb7bd5c3b, 0xc0ba6cad,
    0xedb88320, 0x9abfb3b6, 0x03b6e20c, 0x74b1d29a, 0xead54739, 0x9dd277af, 0x04db2615, 0x73dc1683,
    0xe3630b12, 0x94643b84, 0x0d6d6a3e, 0x7a6a5aa8, 0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1,
    0xf00f9344, 0x8708a3d2, 0x1e01f268, 0x6906c2fe, 0xf762575d, 0x806567cb, 0x196c3671, 0x6e6b06e7,
    0xfed41b76, 0x89d32be0, 0x10da7a5a, 0x67dd4acc, 0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5,
    0xd6d6a3e8, 0xa1d1937e, 0x38d8c2c4, 0x4fdff252, 0xd1bb67f1, 0xa6bc5767, 0x3fb506dd, 0x48b2364b,
    0xd80d2bda, 0xaf0a1b4c, 0x36034af6, 0x41047a60, 0xdf60efc3, 0xa867df55, 0x316e8eef, 0x4669be79,
    0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236, 0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f,
    0xc5ba3bbe, 0xb2bd0b28, 0x2bb45a92, 0x5cb36a04, 0xc2d7ffa7, 0xb5d0cf31, 0x2cd99e8b, 0x5bdeae1d,
    0x9b64c2b0, 0xec63f226, 0x756aa39c, 0x026d930a, 0x9c0906a9, 0xeb0e363f, 0x72076785, 0x05005713,
    0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38, 0x92d28e9b, 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21,
    0x86d3d2d4, 0xf1d4e242, 0x68ddb3f8, 0x1fda836e, 0x81be16cd, 0xf6b9265b, 0x6fb077e1, 0x18b74777,
    0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c, 0x8f659eff, 0xf862ae69, 0x616bffd3, 0x166ccf45,
    0xa00ae278, 0xd70dd2ee, 0x4e048354, 0x3903b3c2, 0xa7672661, 0xd06016f7, 0x4969474d, 0x3e6e77db,
    0xaed16a4a, 0xd9d65adc, 0x40df0b66, 0x37d83bf0, 0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9,
    0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6, 0xbad03605, 0xcdd70693, 0x54de5729, 0x23d967bf,
    0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94, 0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d
};
44 45 46

QGCUASFileManager::QGCUASFileManager(QObject* parent, UASInterface* uas) :
    QObject(parent),
Lorenz Meier's avatar
Lorenz Meier committed
47 48 49 50 51 52 53
    _mav(uas),
    _encdata_seq(0)
{
}

quint32
QGCUASFileManager::crc32(const uint8_t *src, unsigned len, unsigned state)
54
{
Lorenz Meier's avatar
Lorenz Meier committed
55 56 57
    for (unsigned i = 0; i < len; i++)
        state = crctab[(state ^ src[i]) & 0xff] ^ (state >> 8);
    return state;
58 59
}

Lorenz Meier's avatar
Lorenz Meier committed
60 61
void QGCUASFileManager::nothingMessage()
{
62 63
    mavlink_message_t message;

Lorenz Meier's avatar
Lorenz Meier committed
64
    RequestHeader hdr;
65 66 67 68 69
    hdr.magic = 'f';
    hdr.session = 0;
    hdr.crc32 = 0;
    hdr.size = 0;
    hdr.crc32 = crc32((uint8_t*)&hdr, sizeof(hdr) + hdr.size, 0);
Lorenz Meier's avatar
Lorenz Meier committed
70 71 72

    mavlink_msg_encapsulated_data_pack(250, 0, &message, _encdata_seq, (uint8_t*)&hdr);

73
    _mav->sendMessage(message);
Lorenz Meier's avatar
Lorenz Meier committed
74 75 76 77 78
    qDebug() << "Sent message!";
}

void QGCUASFileManager::receiveMessage(LinkInterface* link, mavlink_message_t message)
{
79 80
    emit statusMessage("msg");

none's avatar
none committed
81
    if (message.msgid != MAVLINK_MSG_ID_ENCAPSULATED_DATA) {
82
        emit statusMessage("not encap data");
none's avatar
none committed
83 84 85 86 87 88 89 90 91 92 93 94
        // wtf, not for us
        return;
    }

    mavlink_encapsulated_data_t data;
    mavlink_msg_encapsulated_data_decode(&message, &data);
    const RequestHeader *hdr = (const RequestHeader *)&data.data[0];
    unsigned seqnr = data.seqnr;

    switch (_current_operation) {
    case kCOIdle:
        // we should not be seeing anything here.. shut the other guy up
95
        emit statusMessage("resetting file transfer session");
none's avatar
none committed
96 97 98 99
        sendReset();
        break;

    case kCOList:
100
        if (hdr->opcode == kRspAck) {
none's avatar
none committed
101
            listDecode(&hdr->data[0], hdr->size);
102 103
        } else {
            emit statusMessage("unexpected opcode in List mode");
none's avatar
none committed
104 105
        }
        break;
106 107 108

    default:
        emit statusMessage("message in unexpected state");
Lorenz Meier's avatar
Lorenz Meier committed
109 110 111 112 113
    }
}

void QGCUASFileManager::listRecursively(const QString &from)
{
none's avatar
none committed
114 115 116 117
    if (_current_operation != kCOIdle) {
        // XXX beep and don't do anything
    }

118 119
    // clear the text widget
    emit resetStatusMessages();
none's avatar
none committed
120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200

    // initialise the lister
    _list_path = from;
    _list_offset = 0;
    _current_operation = kCOList;

    // and send the initial request
    sendList();
}

void QGCUASFileManager::listDecode(const uint8_t *data, unsigned len)
{
    unsigned offset = 0;
    unsigned files = 0;

    // parse filenames out of the buffer
    while (offset < len) {

        // get the length of the name
        unsigned nlen = strnlen((const char *)data + offset, len - offset);
        if (nlen == 0) {
            break;
        }

        // put it in the view
        emit statusMessage(QString((const char *)data + offset));

        // account for the name + NUL
        offset += nlen + 1;

        // account for the file
        files++;
    }

    // we have run out of files to list
    if (files == 0) {
        _current_operation = kCOIdle;
    } else {
        // update our state
        _list_offset += files;

        // and send another request
        sendList();
    }
}

void QGCUASFileManager::sendReset()
{
    mavlink_message_t message;

    RequestHeader hdr;
    hdr.magic = 'f';
    hdr.session = 0;
    hdr.opcode = kCmdReset;
    hdr.crc32 = 0;
    hdr.offset = 0;
    hdr.size = 0;
    hdr.crc32 = crc32((uint8_t*)&hdr, sizeof(hdr) + hdr.size, 0);

    mavlink_msg_encapsulated_data_pack(250, 0, &message, _encdata_seq, (uint8_t*)&hdr); // XXX 250 is a magic length

    _mav->sendMessage(message);
}

void QGCUASFileManager::sendList()
{
    mavlink_message_t message;

    RequestHeader hdr;
    hdr.magic = 'f';
    hdr.session = 0;
    hdr.opcode = kCmdList;
    hdr.crc32 = 0;
    hdr.offset = _list_offset;
    strncpy((char *)&hdr.data[0], _list_path.toStdString().c_str(), 200); // XXX should have a real limit here
    hdr.size = strlen((const char *)&hdr.data[0]);

    hdr.crc32 = crc32((uint8_t*)&hdr, sizeof(hdr) + hdr.size, 0);

    mavlink_msg_encapsulated_data_pack(250, 0, &message, _encdata_seq, (uint8_t*)&hdr); // XXX 250 is a magic length

201 202
    emit statusMessage("sending List request...");

none's avatar
none committed
203
    _mav->sendMessage(message);
Lorenz Meier's avatar
Lorenz Meier committed
204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224
}

void QGCUASFileManager::downloadPath(const QString &from, const QString &to)
{

    // Send path, e.g. /fs/microsd and download content
    // recursively into a local directory

    char buf[255];
    unsigned len = 10;

    QByteArray data(buf, len);
    QString filename = "log001.bin"; // XXX get this from onboard

    // Qt takes care of slash conversions in paths
    QFile file(to + QDir::separator() + filename);
    file.open(QIODevice::WriteOnly);
    file.write(data);
    file.close();

    emit statusMessage(QString("Downloaded: %1 to directory %2").arg(filename).arg(to));
225
}