main.cc 8.36 KB
Newer Older
1 2 3 4 5 6 7 8
/****************************************************************************
 *
 *   (c) 2009-2016 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>
 *
 * QGroundControl is licensed according to the terms in the file
 * COPYING.md in the root of the source code directory.
 *
 ****************************************************************************/
pixhawk's avatar
pixhawk committed
9 10 11 12 13 14 15 16 17


/**
 * @file
 *   @brief Main executable
 *   @author Lorenz Meier <mavteam@student.ethz.ch>
 *
 */

18
#include <QtGlobal>
19
#include <QApplication>
20
#include <QIcon>
Don Gagne's avatar
Don Gagne committed
21
#include <QSslSocket>
22
#include <QProcessEnvironment>
23 24
#include <QHostAddress>
#include <QUdpSocket>
Don Gagne's avatar
Don Gagne committed
25
#include <QtPlugin>
26
#include <QStringListModel>
Don Gagne's avatar
Don Gagne committed
27
#include "QGCApplication.h"
28
#include "AppMessages.h"
29

30 31
#ifndef __mobile__
    #include "QGCSerialPortInfo.h"
32
    #include "RunGuard.h"
33 34
#endif

Valentin Platzgummer's avatar
Valentin Platzgummer committed
35 36 37 38 39 40 41 42
#ifndef QML_PROFILE
#define QML_PROFILE 1
#endif

#if QML_PROFILE
#include <QQmlDebuggingEnabler>
#endif

43 44 45 46
#ifdef UNITTEST_BUILD
    #include "UnitTest.h"
#endif

47
#ifdef QT_DEBUG
48 49 50 51
    #include "CmdLineOptParser.h"
    #ifdef Q_OS_WIN
        #include <crtdbg.h>
    #endif
52
#endif
53

dogmaphobic's avatar
dogmaphobic committed
54
#ifdef QGC_ENABLE_BLUETOOTH
dogmaphobic's avatar
dogmaphobic committed
55 56 57
#include <QtBluetooth/QBluetoothSocket>
#endif

Don Gagne's avatar
Don Gagne committed
58
#include <iostream>
dogmaphobic's avatar
dogmaphobic committed
59
#include "QGCMapEngine.h"
unknown's avatar
unknown committed
60

61 62 63
// Snake
#include "Wima/WimaControllerDetail.h"

64
/* SDL does ugly things to main() */
65
#ifdef main
66
#undef main
67
#endif
68

69
#ifndef __mobile__
70
#ifndef NO_SERIAL_LINK
71
    Q_DECLARE_METATYPE(QGCSerialPortInfo)
72
#endif
73
#endif
74 75

#ifdef Q_OS_WIN
76 77 78

#include <windows.h>

79 80 81 82 83
/// @brief CRT Report Hook installed using _CrtSetReportHook. We install this hook when
/// we don't want asserts to pop a dialog on windows.
int WindowsCrtReportHook(int reportType, char* message, int* returnValue)
{
    Q_UNUSED(reportType);
84

85 86 87 88 89
    std::cerr << message << std::endl;  // Output message to stderr
    *returnValue = 0;                   // Don't break into debugger
    return true;                        // We handled this fully ourselves
}

90 91
#endif

Gus Grubba's avatar
Gus Grubba committed
92
#if defined(__android__) && !defined(NO_SERIAL_LINK)
Don Gagne's avatar
Don Gagne committed
93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110
#include <jni.h>
#include "qserialport.h"

jint JNI_OnLoad(JavaVM* vm, void* reserved)
{
    Q_UNUSED(reserved);

    JNIEnv* env;
    if (vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6) != JNI_OK) {
        return -1;
    }

    QSerialPort::setNativeMethods();

    return JNI_VERSION_1_6;
}
#endif

111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
#ifdef __android__
#include <QtAndroid>
bool checkAndroidWritePermission() {
    QtAndroid::PermissionResult r = QtAndroid::checkPermission("android.permission.WRITE_EXTERNAL_STORAGE");
    if(r == QtAndroid::PermissionResult::Denied) {
        QtAndroid::requestPermissionsSync( QStringList() << "android.permission.WRITE_EXTERNAL_STORAGE" );
        r = QtAndroid::checkPermission("android.permission.WRITE_EXTERNAL_STORAGE");
        if(r == QtAndroid::PermissionResult::Denied) {
             return false;
        }
   }
   return true;
}
#endif

pixhawk's avatar
pixhawk committed
126 127 128 129 130 131 132
/**
 * @brief Starts the application
 *
 * @param argc Number of commandline arguments
 * @param argv Commandline arguments
 * @return exit code, 0 for normal exit and !=0 for error cases
 */
133

pixhawk's avatar
pixhawk committed
134 135
int main(int argc, char *argv[])
{
136 137 138 139 140 141 142
#ifndef __mobile__
    RunGuard guard("QGroundControlRunGuardKey");
    if (!guard.tryToRun()) {
        return 0;
    }
#endif

143
#ifdef Q_OS_UNIX
Daniel Agar's avatar
Daniel Agar committed
144
    //Force writing to the console on UNIX/BSD devices
145 146 147 148
    if (!qEnvironmentVariableIsSet("QT_LOGGING_TO_CONSOLE"))
        qputenv("QT_LOGGING_TO_CONSOLE", "1");
#endif

149
    // install the message handler
150
    AppMessages::installHandler();
151
QList<QPair<QByteArray,QByteArray>
Lorenz Meier's avatar
Lorenz Meier committed
152
#ifdef Q_OS_MAC
153
#ifndef __ios__
154 155 156
    // Prevent Apple's app nap from screwing us over
    // tip: the domain can be cross-checked on the command line with <defaults domains>
    QProcess::execute("defaults write org.qgroundcontrol.qgroundcontrol NSAppSleepDisabled -bool YES");
157
#endif
Lorenz Meier's avatar
Lorenz Meier committed
158
#endif
159

160
#ifdef Q_OS_WIN
161 162
    // Set our own OpenGL buglist
    qputenv("QT_OPENGL_BUGLIST", ":/opengl/resources/opengl/buglist.json");
163 164 165 166 167 168 169 170 171 172 173

    // Allow for command line override of renderer
    for (int i = 0; i < argc; i++) {
        const QString arg(argv[i]);
        if (arg == QStringLiteral("-angle")) {
            QCoreApplication::setAttribute(Qt::AA_UseOpenGLES);
            break;
        } else if (arg == QStringLiteral("-swrast")) {
            QCoreApplication::setAttribute(Qt::AA_UseSoftwareOpenGL);
            break;
        }
174
    }
175
#endif
176 177 178 179 180

    // The following calls to qRegisterMetaType are done to silence debug output which warns
    // that we use these types in signals, and without calling qRegisterMetaType we can't queue
    // these signals. In general we don't queue these signals, but we do what the warning says
    // anyway to silence the debug output.
Gus Grubba's avatar
Gus Grubba committed
181
#ifndef NO_SERIAL_LINK
182
    qRegisterMetaType<QSerialPort::SerialPortError>();
dogmaphobic's avatar
dogmaphobic committed
183
#endif
dogmaphobic's avatar
dogmaphobic committed
184
#ifdef QGC_ENABLE_BLUETOOTH
dogmaphobic's avatar
dogmaphobic committed
185 186
    qRegisterMetaType<QBluetoothSocket::SocketError>();
    qRegisterMetaType<QBluetoothServiceInfo>();
dogmaphobic's avatar
dogmaphobic committed
187
#endif
188
    qRegisterMetaType<QAbstractSocket::SocketError>();
189
#ifndef __mobile__
190
#ifndef NO_SERIAL_LINK
191
    qRegisterMetaType<QGCSerialPortInfo>();
192
#endif
193
#endif
dogmaphobic's avatar
dogmaphobic committed
194 195

    // We statically link our own QtLocation plugin
196 197 198 199 200 201

#ifdef Q_OS_WIN
    // In Windows, the compiler doesn't see the use of the class created by Q_IMPORT_PLUGIN
#pragma warning( disable : 4930 4101 )
#endif

202
    Q_IMPORT_PLUGIN(QGeoServiceProviderFactoryQGC)
203

Don Gagne's avatar
Don Gagne committed
204
    bool runUnitTests = false;          // Run unit tests
205

206
#ifdef QT_DEBUG
207
    // We parse a small set of command line options here prior to QGCApplication in order to handle the ones
208
    // which need to be handled before a QApplication object is started.
209

Don Gagne's avatar
Don Gagne committed
210
    bool stressUnitTests = false;       // Stress test unit tests
211
    bool quietWindowsAsserts = false;   // Don't let asserts pop dialog boxes
212

213
    QString unitTestOptions;
214
    CmdLineOpt_t rgCmdLineOptions[] = {
215
        { "--unittest",             &runUnitTests,          &unitTestOptions },
216
        { "--unittest-stress",      &stressUnitTests,       &unitTestOptions },
217
        { "--no-windows-assert-ui", &quietWindowsAsserts,   NULL },
218 219
        // Add additional command line option flags here
    };
220

221
    ParseCmdLineOptions(argc, argv, rgCmdLineOptions, sizeof(rgCmdLineOptions)/sizeof(rgCmdLineOptions[0]), false);
222 223 224
    if (stressUnitTests) {
        runUnitTests = true;
    }
225

226 227 228 229 230
    if (quietWindowsAsserts) {
#ifdef Q_OS_WIN
        _CrtSetReportHook(WindowsCrtReportHook);
#endif
    }
231 232 233 234 235 236 237 238

#ifdef Q_OS_WIN
    if (runUnitTests) {
        // Don't pop up Windows Error Reporting dialog when app crashes. This prevents TeamCity from
        // hanging.
        DWORD dwMode = SetErrorMode(SEM_NOGPFAULTERRORBOX);
        SetErrorMode(dwMode | SEM_NOGPFAULTERRORBOX);
    }
239
#endif
240
#endif // QT_DEBUG
241

Don Gagne's avatar
Don Gagne committed
242
    QGCApplication* app = new QGCApplication(argc, argv, runUnitTests);
243
    Q_CHECK_PTR(app);
244

245 246 247 248
#ifdef Q_OS_LINUX
    QApplication::setWindowIcon(QIcon(":/res/resources/icons/qgroundcontrol.ico"));
#endif /* Q_OS_LINUX */

249 250 251 252 253 254
    // There appears to be a threading issue in qRegisterMetaType which can cause it to throw a qWarning
    // about duplicate type converters. This is caused by a race condition in the Qt code. Still working
    // with them on tracking down the bug. For now we register the type which is giving us problems here
    // while we only have the main thread. That should prevent it from hitting the race condition later
    // on in the code.
    qRegisterMetaType<QList<QPair<QByteArray,QByteArray> > >();
255

256 257 258
    // Snake
    qRegisterMetaType<WorkerResult>("WorkerResult");

259
    app->_initCommon();
dogmaphobic's avatar
dogmaphobic committed
260 261
    //-- Initialize Cache System
    getQGCMapEngine()->init();
262

Valentin Platzgummer's avatar
Valentin Platzgummer committed
263 264 265 266 267

#if QML_PROFILE
   QQmlDebuggingEnabler enabler;
#endif

268
    int exitCode = 0;
269

270
#ifdef UNITTEST_BUILD
271
    if (runUnitTests) {
272 273 274 275 276 277 278 279 280 281 282 283 284 285 286
        for (int i=0; i < (stressUnitTests ? 20 : 1); i++) {
            if (!app->_initForUnitTests()) {
                return -1;
            }

            // Run the test
            int failures = UnitTest::run(unitTestOptions);
            if (failures == 0) {
                qDebug() << "ALL TESTS PASSED";
                exitCode = 0;
            } else {
                qDebug() << failures << " TESTS FAILED!";
                exitCode = -failures;
                break;
            }
287
        }
288
    } else
289
#endif
290
    {
291 292 293 294

#ifdef __android__
        checkAndroidWritePermission();
#endif
295 296 297
        if (!app->_initForNormalAppBoot()) {
            return -1;
        }
298

Don Gagne's avatar
Don Gagne committed
299
        exitCode = app->exec();
300
    }
301

Don Gagne's avatar
Don Gagne committed
302
    app->_shutdown();
Don Gagne's avatar
Don Gagne committed
303
    delete app;
dogmaphobic's avatar
dogmaphobic committed
304 305
    //-- Shutdown Cache System
    destroyMapEngine();
306

Don Gagne's avatar
Don Gagne committed
307
    qDebug() << "After app delete";
308

Don Gagne's avatar
Don Gagne committed
309
    return exitCode;
pixhawk's avatar
pixhawk committed
310
}