mirror of https://github.com/nymea/nymea-gpio
174 lines
7.2 KiB
C++
174 lines
7.2 KiB
C++
/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
|
|
*
|
|
* Copyright 2013 - 2020, nymea GmbH
|
|
* Contact: contact@nymea.io
|
|
*
|
|
* This file is part of nymea.
|
|
* This project including source code and documentation is protected by
|
|
* copyright law, and remains the property of nymea GmbH. All rights, including
|
|
* reproduction, publication, editing and translation, are reserved. The use of
|
|
* this project is subject to the terms of a license agreement to be concluded
|
|
* with nymea GmbH in accordance with the terms of use of nymea GmbH, available
|
|
* under https://nymea.io/license
|
|
*
|
|
* GNU General Public License Usage
|
|
* Alternatively, this project may be redistributed and/or modified under the
|
|
* terms of the GNU General Public License as published by the Free Software
|
|
* Foundation, GNU version 3. This project 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 project. If not, see <https://www.gnu.org/licenses/>.
|
|
*
|
|
* For any further details and any questions please contact us under
|
|
* contact@nymea.io or see our FAQ/Licensing Information on
|
|
* https://nymea.io/license/faq
|
|
*
|
|
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
|
|
|
#include <QCommandLineParser>
|
|
|
|
#include "application.h"
|
|
#include "gpiomonitor.h"
|
|
|
|
int main(int argc, char *argv[])
|
|
{
|
|
Application application(argc, argv);
|
|
application.setOrganizationName("guh");
|
|
application.setApplicationName("nymea-gpio-tool");
|
|
application.setApplicationVersion(VERSION_STRING);
|
|
|
|
QCommandLineParser parser;
|
|
parser.addHelpOption();
|
|
parser.addVersionOption();
|
|
QString applicationDescription = QString("\nnymea-gpio-tool is a command line tool which allowes to interact with GPIOs.\n"
|
|
"Version: %1\n"
|
|
"Copyright %2 2019 Simon Stürz <simon.stuerz@nymea.io>\n\n"
|
|
"Released under the GNU GENERAL PUBLIC LICENSE Version 3.\n").arg(application.applicationVersion()).arg(QChar(0xA9));
|
|
parser.setApplicationDescription(applicationDescription);
|
|
|
|
QCommandLineOption gpioOption(QStringList() << "g" << "gpio", "The gpio number to use.", "GPIO");
|
|
parser.addOption(gpioOption);
|
|
|
|
QCommandLineOption interruptOption(QStringList() << "i" << "interrupt", "Configure the input GPIO to the given interrupt. This option is only allowed for monitoring. Allowerd interrupts are: [rising, falling, both, none]. Default is \"both\".", "INTERRUPT");
|
|
parser.addOption(interruptOption);
|
|
|
|
QCommandLineOption valueOption(QStringList() << "s" << "set-value", "Configure the GPIO to output and set the value. Allowerd values are: [0, 1].", "VALUE");
|
|
parser.addOption(valueOption);
|
|
|
|
QCommandLineOption monitorOption(QStringList() << "m" << "monitor", "Monitor the given GPIO. The GPIO will automatically configured as input and print any change regarding to the given interrupt behaviour.");
|
|
parser.addOption(monitorOption);
|
|
|
|
QCommandLineOption activeLowOption(QStringList() << "l" << "active-low", "Configure the pin as active low (default is active high).");
|
|
parser.addOption(activeLowOption);
|
|
|
|
parser.process(application);
|
|
|
|
// Make sure there is a GPIO number passed
|
|
if (!parser.isSet(gpioOption)) {
|
|
qCritical() << "No GPIO number specified. Please specify a valid GPIO number using -g, --gpio GPIO";
|
|
parser.showHelp(EXIT_FAILURE);
|
|
}
|
|
|
|
// Verify GPIO number
|
|
bool gpioNumberOk;
|
|
int gpioNumber = parser.value(gpioOption).toInt(&gpioNumberOk);
|
|
if (!gpioNumberOk || gpioNumber < 0) {
|
|
qCritical() << "Invalid GPIO number" << parser.value(gpioOption) << "passed. The GPIO number has to be a positiv integer.";
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
// Verify input output operations
|
|
if ((parser.isSet(interruptOption) || parser.isSet(monitorOption)) && parser.isSet(valueOption)) {
|
|
qCritical() << "Invalid parameter combination. The set value can only be used for output GPIO, the monitor and interrupt parameter can only be used for input GPIO.";
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
Gpio::Edge edge = Gpio::EdgeBoth;
|
|
if (parser.isSet(interruptOption)) {
|
|
if (parser.value(interruptOption).toLower() == "rising") {
|
|
edge = Gpio::EdgeRising;
|
|
} else if (parser.value(interruptOption).toLower() == "falling") {
|
|
edge = Gpio::EdgeFalling;
|
|
} else if (parser.value(interruptOption).toLower() == "none") {
|
|
edge = Gpio::EdgeNone;
|
|
} else if (parser.value(interruptOption).toLower() == "both") {
|
|
edge = Gpio::EdgeBoth;
|
|
} else {
|
|
qCritical() << "Invalid interrupt parameter" << parser.value(interruptOption) << "passed. Valid options are [rising, falling, both, none].";
|
|
return EXIT_FAILURE;
|
|
}
|
|
}
|
|
|
|
bool activeLow = parser.isSet(activeLowOption);
|
|
|
|
Gpio::Value value = Gpio::ValueInvalid;
|
|
if (parser.isSet(valueOption)) {
|
|
if (parser.value(valueOption) == "1") {
|
|
value = Gpio::ValueHigh;
|
|
} else if (parser.value(valueOption) == "0") {
|
|
value = Gpio::ValueLow;
|
|
} else {
|
|
qCritical() << "Invalid set value parameter" << parser.value(valueOption) << "passed. Valid options are [0, 1].";
|
|
return EXIT_FAILURE;
|
|
}
|
|
}
|
|
|
|
if (!Gpio::isAvailable()) {
|
|
qCritical() << "There are no GPIOs available on this platform.";
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
// Configure the GPIO
|
|
if (parser.isSet(valueOption)) {
|
|
Gpio *gpio = new Gpio(gpioNumber);
|
|
if (!gpio->exportGpio()) {
|
|
qCritical() << "Could not export GPIO" << gpioNumber;
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
if (!gpio->setDirection(Gpio::DirectionOutput)) {
|
|
qCritical() << "Could not configure GPIO" << gpioNumber << "as output.";
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
if (parser.isSet(activeLowOption)) {
|
|
if (!gpio->setActiveLow(activeLow)) {
|
|
qCritical() << "Could not set GPIO" << gpioNumber << "to active low" << activeLow;
|
|
return EXIT_FAILURE;
|
|
}
|
|
}
|
|
|
|
// Finally set the value
|
|
if (!gpio->setValue(value)) {
|
|
qCritical() << "Could not set GPIO" << gpioNumber << "value to" << value;
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
delete gpio;
|
|
return EXIT_SUCCESS;
|
|
} else {
|
|
GpioMonitor *monitor = new GpioMonitor(gpioNumber);
|
|
|
|
// Inform about interrupt
|
|
QObject::connect(monitor, &GpioMonitor::valueChanged, [gpioNumber](bool value) {
|
|
qDebug() << "GPIO" << gpioNumber << "value changed:" << (value ? "1" : "0");
|
|
});
|
|
|
|
// Enable the monitor
|
|
if (!monitor->enable(activeLow, edge)) {
|
|
qCritical() << "Could not enable GPIO" << gpioNumber << "monitor.";
|
|
return EXIT_FAILURE;
|
|
}
|
|
|
|
// Clean up the gpio once done
|
|
QObject::connect(&application, &Application::aboutToQuit, [monitor](){
|
|
delete monitor;
|
|
});
|
|
}
|
|
|
|
return application.exec();
|
|
}
|