├── images
├── file.png
├── home.png
├── tick.png
├── web.png
├── wifi.png
├── error.png
├── refresh.png
├── users.png
├── wifi0.png
├── wifi1.png
├── wifi2.png
├── wifi3.png
├── wifi4.png
├── bell-gray.png
├── calendar.png
├── settings.png
├── stopwatch.png
├── up-white.png
├── wifi-gray.png
├── down-white.png
├── home-small.png
├── photo-camera.png
├── placeholder.png
├── video-player.png
├── left-arrow-gray.png
└── right-arrow-gray.png
├── README.md
├── qml.qrc
├── wifimanager.h
├── wifi
├── wifiinfo.cpp
├── wifiinfo.h
├── wifi_scan.h
└── wifi_scan.c
├── TimeRect.qml
├── .gitignore
├── main.cpp
├── qmlbridge.cpp
├── Button.qml
├── images.qrc
├── qmlbridge.h
├── PageManager.qml
├── NotifyButton.qml
├── Checkbox.qml
├── WifiButton.qml
├── BlApp.pro
├── WebDashboard.qml
├── wifimanager.cpp
├── Home.qml
├── WifiConfig.qml
└── LICENSE
/images/file.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/file.png
--------------------------------------------------------------------------------
/images/home.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/home.png
--------------------------------------------------------------------------------
/images/tick.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/tick.png
--------------------------------------------------------------------------------
/images/web.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/web.png
--------------------------------------------------------------------------------
/images/wifi.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/wifi.png
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # pidash
2 | Qt, QML, C++ dashboard UI for raspberry pi management etc.
3 |
--------------------------------------------------------------------------------
/images/error.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/error.png
--------------------------------------------------------------------------------
/images/refresh.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/refresh.png
--------------------------------------------------------------------------------
/images/users.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/users.png
--------------------------------------------------------------------------------
/images/wifi0.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/wifi0.png
--------------------------------------------------------------------------------
/images/wifi1.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/wifi1.png
--------------------------------------------------------------------------------
/images/wifi2.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/wifi2.png
--------------------------------------------------------------------------------
/images/wifi3.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/wifi3.png
--------------------------------------------------------------------------------
/images/wifi4.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/wifi4.png
--------------------------------------------------------------------------------
/images/bell-gray.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/bell-gray.png
--------------------------------------------------------------------------------
/images/calendar.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/calendar.png
--------------------------------------------------------------------------------
/images/settings.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/settings.png
--------------------------------------------------------------------------------
/images/stopwatch.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/stopwatch.png
--------------------------------------------------------------------------------
/images/up-white.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/up-white.png
--------------------------------------------------------------------------------
/images/wifi-gray.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/wifi-gray.png
--------------------------------------------------------------------------------
/images/down-white.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/down-white.png
--------------------------------------------------------------------------------
/images/home-small.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/home-small.png
--------------------------------------------------------------------------------
/images/photo-camera.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/photo-camera.png
--------------------------------------------------------------------------------
/images/placeholder.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/placeholder.png
--------------------------------------------------------------------------------
/images/video-player.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/video-player.png
--------------------------------------------------------------------------------
/images/left-arrow-gray.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/left-arrow-gray.png
--------------------------------------------------------------------------------
/images/right-arrow-gray.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/stoic1979/pidash/HEAD/images/right-arrow-gray.png
--------------------------------------------------------------------------------
/qml.qrc:
--------------------------------------------------------------------------------
1 |
2 |
3 | Home.qml
4 | Button.qml
5 | WebDashboard.qml
6 | WifiConfig.qml
7 | PageManager.qml
8 | NotifyButton.qml
9 | TimeRect.qml
10 | WifiButton.qml
11 | Checkbox.qml
12 |
13 |
14 |
--------------------------------------------------------------------------------
/wifimanager.h:
--------------------------------------------------------------------------------
1 | #ifndef WIFIMANAGER_H
2 | #define WIFIMANAGER_H
3 |
4 | #include
5 |
6 | class WifiManager : public QObject
7 | {
8 | Q_OBJECT
9 | public:
10 | explicit WifiManager(QObject *parent = 0);
11 |
12 | QList getWifiNetworks();
13 | void connectToWifi(QString name, QString pass);
14 |
15 | signals:
16 |
17 | public slots:
18 |
19 |
20 |
21 | private:
22 | QList dataList;
23 | };
24 |
25 | #endif // WIFIMANAGER_H
26 |
--------------------------------------------------------------------------------
/wifi/wifiinfo.cpp:
--------------------------------------------------------------------------------
1 | #include "wifiinfo.h"
2 |
3 | WifiInfo::WifiInfo(QString name, int strength, QObject *parent):
4 | name(name),
5 | strength(strength),
6 | QObject(parent)
7 | {}
8 |
9 |
10 | QString WifiInfo::getName() {
11 | return name;
12 | }
13 |
14 | int WifiInfo::getStrength() {
15 | return strength;
16 | }
17 |
18 | void WifiInfo::setName(QString n) {
19 | name = n;
20 | }
21 |
22 | void WifiInfo::setStrength(int s) {
23 | strength = s;
24 | }
25 |
--------------------------------------------------------------------------------
/TimeRect.qml:
--------------------------------------------------------------------------------
1 | import QtQuick 2.0
2 |
3 | Rectangle {
4 |
5 | property alias textColor: txt.color
6 |
7 | Text {
8 | id: txt
9 | text: "123"
10 | anchors.centerIn: parent
11 | }
12 | Timer {
13 | id: textTimer
14 | interval: 1000
15 | repeat: true
16 | running: true
17 | triggeredOnStart: true
18 | onTriggered: txt.text = Qt.formatTime(new Date(),"hh:mm:ss")
19 | }
20 |
21 | }
22 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # C++ objects and libs
2 |
3 | *.slo
4 | *.lo
5 | *.o
6 | *.a
7 | *.la
8 | *.lai
9 | *.so
10 | *.dll
11 | *.dylib
12 |
13 | # Qt-es
14 |
15 | /.qmake.cache
16 | /.qmake.stash
17 | *.pro.user
18 | *.pro.user.*
19 | *.qbs.user
20 | *.qbs.user.*
21 | *.moc
22 | moc_*.cpp
23 | qrc_*.cpp
24 | ui_*.h
25 | Makefile*
26 | *build-*
27 |
28 | # QtCreator
29 |
30 | *.autosave
31 |
32 | # QtCtreator Qml
33 | *.qmlproject.user
34 | *.qmlproject.user.*
35 |
36 | # QtCtreator CMake
37 | CMakeLists.txt.user
38 |
39 |
--------------------------------------------------------------------------------
/main.cpp:
--------------------------------------------------------------------------------
1 | #include
2 | #include
3 | #include
4 | #include
5 | #include
6 | #include
7 | #include
8 |
9 | #include "qmlbridge.h"
10 |
11 | #include "utils.h"
12 |
13 | int main(int argc, char *argv[])
14 | {
15 | QGuiApplication app(argc, argv);
16 |
17 | QtWebView::initialize();
18 |
19 | QmlBridge qmlBridge;
20 |
21 |
22 | Utils utils;
23 | qDebug() << "Result: " << utils.getCmdResult("ls");
24 |
25 |
26 | QQmlApplicationEngine engine;
27 |
28 | engine.rootContext()->setContextProperty("qmlBridge", &qmlBridge);
29 | engine.load(QUrl(QStringLiteral("qrc:/PageManager.qml")));
30 |
31 | return app.exec();
32 | }
33 |
--------------------------------------------------------------------------------
/wifi/wifiinfo.h:
--------------------------------------------------------------------------------
1 | #ifndef WIFIINFO_H
2 | #define WIFIINFO_H
3 |
4 | #include
5 |
6 | class WifiInfo : public QObject
7 | {
8 | Q_OBJECT
9 |
10 | Q_PROPERTY(QString name READ getName WRITE setName)
11 | //Q_PROPERTY(QString name READ name WRITE setName NOTIFY nameChanged)
12 | Q_PROPERTY(int strength READ getStrength WRITE setStrength)
13 | //Q_PROPERTY(int strength READ strength WRITE setstrength NOTIFY strengthChanged)
14 |
15 | public:
16 | explicit WifiInfo(QString name, int strength, QObject *parent = 0);
17 |
18 | Q_INVOKABLE QString getName();
19 | Q_INVOKABLE int getStrength();
20 | Q_INVOKABLE void setName(QString n);
21 | Q_INVOKABLE void setStrength(int s);
22 |
23 |
24 | signals:
25 |
26 | public slots:
27 |
28 | private:
29 | QString name;
30 | int strength;
31 | };
32 |
33 | #endif // WIFIINFO_H
34 |
--------------------------------------------------------------------------------
/qmlbridge.cpp:
--------------------------------------------------------------------------------
1 | #include
2 | #include "qmlbridge.h"
3 | #include "wifimanager.h"
4 | #include "utils.h"
5 |
6 |
7 | QmlBridge::QmlBridge(QObject *parent) : QObject(parent), wifiCount(0)
8 | {}
9 |
10 |
11 | void QmlBridge::test() {
12 | qDebug() << "[QmlBridge] :: test";
13 | }
14 |
15 | bool QmlBridge::postMessage(const QString &msg) {
16 | qDebug() << "[QmlBridge] :: Called the C++ method with" << msg;
17 | return true;
18 | }
19 |
20 | QList QmlBridge::getWifiNetworks() {
21 | qDebug() << "[QmlBridge] :: getWifiNetworks";
22 |
23 | WifiManager wm;
24 | return wm.getWifiNetworks();
25 | }
26 |
27 | int QmlBridge::getWifisCount() {
28 | return wifiCount;
29 | }
30 |
31 | QString QmlBridge::getConectedWifiName() {
32 | Utils utils;
33 | return utils.getCmdResult("iwgetid -r");
34 | }
35 |
36 | void QmlBridge::connectToWifi(QString name, QString pass) {
37 | WifiManager wm;
38 | wm.connectToWifi(name, pass);
39 | }
40 |
41 | QString QmlBridge::getDeivceName() {
42 | Utils utils;
43 | return utils.getCmdResult("whoami");
44 | }
45 |
--------------------------------------------------------------------------------
/Button.qml:
--------------------------------------------------------------------------------
1 | import QtQuick 2.0
2 |
3 | Rectangle {
4 | id: button
5 |
6 | signal clicked
7 | property alias imgSrc: img.source
8 | property alias text: txt.text
9 | property alias textColor: txt.color
10 | property alias imgWidth: img.width
11 | property alias imgHeight: img.height
12 |
13 |
14 | color: "#444444"
15 | //color: "#dddddd"
16 | //color: "#2E72EF"
17 | radius: 4
18 | border.color: "#cccccc"
19 |
20 | Image {
21 | id: img
22 | anchors.centerIn: parent
23 | width: button.width * 3/4
24 | height: button.height * 3/4
25 | }
26 |
27 | Text {
28 | id: txt
29 | anchors.centerIn: button
30 | }
31 |
32 | MouseArea {
33 | id: mouseArea
34 | enabled: button.enabled
35 | anchors.fill: button
36 | hoverEnabled: true
37 |
38 | onClicked: button.clicked()
39 |
40 | onPressed: {
41 | button.opacity = 0.5
42 | }
43 |
44 | onReleased: {
45 | button.opacity = 1.0
46 | }
47 |
48 | }
49 |
50 | }//button
51 |
--------------------------------------------------------------------------------
/images.qrc:
--------------------------------------------------------------------------------
1 |
2 |
3 | images/web.png
4 | images/wifi-gray.png
5 | images/bell-gray.png
6 | images/left-arrow-gray.png
7 | images/right-arrow-gray.png
8 | images/settings.png
9 | images/calendar.png
10 | images/file.png
11 | images/photo-camera.png
12 | images/placeholder.png
13 | images/stopwatch.png
14 | images/users.png
15 | images/video-player.png
16 | images/wifi.png
17 | images/home-small.png
18 | images/up-white.png
19 | images/down-white.png
20 | images/wifi0.png
21 | images/wifi1.png
22 | images/wifi2.png
23 | images/wifi3.png
24 | images/wifi4.png
25 | images/refresh.png
26 | images/tick.png
27 |
28 |
29 |
--------------------------------------------------------------------------------
/qmlbridge.h:
--------------------------------------------------------------------------------
1 | #ifndef QMLBRIDGE_H
2 | #define QMLBRIDGE_H
3 |
4 | #include
5 | #include
6 |
7 | #include
8 |
9 |
10 | class QmlBridge : public QObject
11 | {
12 | Q_OBJECT
13 | public:
14 | explicit QmlBridge(QObject *parent = 0);
15 |
16 | Q_INVOKABLE void test();
17 |
18 | Q_INVOKABLE bool postMessage(const QString &msg);
19 |
20 | Q_INVOKABLE QList getWifiNetworks();
21 | Q_INVOKABLE int getWifisCount();
22 |
23 | Q_INVOKABLE QString getConectedWifiName();
24 |
25 | Q_INVOKABLE void connectToWifi(QString name, QString pass);
26 |
27 | Q_INVOKABLE QString getDeivceName();
28 |
29 | signals:
30 |
31 | public slots:
32 | void cppSlot(const QVariant &v) {
33 | qDebug() << "Called the C++ slot with value:" << v;
34 |
35 | QQuickItem *item =
36 | qobject_cast(v.value());
37 | qDebug() << "Item dimensions:" << item->width()
38 | << item->height();
39 | }
40 |
41 | private:
42 | int wifiCount;
43 | QStringListModel model;
44 | QStringList list;
45 |
46 | };
47 |
48 | #endif // QMLBRIDGE_H
49 |
--------------------------------------------------------------------------------
/PageManager.qml:
--------------------------------------------------------------------------------
1 | import QtQuick 2.7
2 | import QtQuick.Window 2.2
3 |
4 |
5 | //-------------------------------------------------------------
6 | //
7 | // Container page with loader to load qml pages dynamically
8 | //
9 | //-------------------------------------------------------------
10 | Window {
11 | id: pageManager
12 | visible: true
13 | width: 800
14 | height: 480
15 |
16 | Loader {
17 | id: pageLoader
18 | anchors.fill: parent
19 | }
20 |
21 | Connections {
22 | target: pageLoader.item
23 |
24 | //------------------------------------------------+
25 | // slot for loading given page |
26 | // to be trigered via signals from various pages |
27 | // to change to some other page |
28 | //------------------------------------------------+
29 | onLoadPage: {
30 | console.log("Loading page: " + page);
31 | pageLoader.source = page;
32 | }
33 | }
34 |
35 | Component.onCompleted: {
36 | // set default page to home page
37 | pageLoader.source = "Home.qml";
38 | }
39 |
40 | }//pageManager
41 |
--------------------------------------------------------------------------------
/NotifyButton.qml:
--------------------------------------------------------------------------------
1 | import QtQuick 2.0
2 |
3 | Rectangle {
4 | id: button
5 |
6 | signal clicked
7 | property alias imgSrc: img.source
8 |
9 | property int count
10 |
11 | color: "white"
12 | radius: 4
13 |
14 | border.color: "#999999"
15 |
16 | Rectangle {
17 | id: rectLeft
18 | width: button.width/2 - 4
19 | height: button.height - 4
20 | x: 2
21 | y: 2
22 |
23 | Image {
24 | id: img
25 | anchors.centerIn: parent
26 | }
27 | }
28 |
29 | Rectangle {
30 | id: rectRight
31 | width: button.width/2 - 4
32 | height: button.height - 4
33 | anchors.left: rectLeft.right
34 | y:2
35 |
36 | Text {
37 | id: txt
38 | text: "" + count
39 | anchors.centerIn: parent
40 | }
41 | }
42 |
43 |
44 | MouseArea {
45 | id: mouseArea
46 | enabled: button.enabled
47 | anchors.fill: button
48 | hoverEnabled: true
49 |
50 | onClicked: button.clicked()
51 |
52 | onPressed: {
53 | button.opacity = 0.5
54 | }
55 |
56 | onReleased: {
57 | button.opacity = 1.0
58 | }
59 |
60 | }
61 |
62 | }//Rectangle
63 |
--------------------------------------------------------------------------------
/Checkbox.qml:
--------------------------------------------------------------------------------
1 | import QtQuick 2.0
2 |
3 | Rectangle {
4 | id: checkbox
5 |
6 | Accessible.role: Accessible.CheckBox
7 |
8 | property string text: "CheckBox"
9 | property bool checked // required variable
10 |
11 | // width, height for checkbox's square rect
12 | property int boxLen: 20
13 |
14 | width: 100
15 | height: 40
16 |
17 | Row {
18 | spacing: 2
19 |
20 | Rectangle {
21 | width: boxLen
22 | height: boxLen
23 | border.width: checkbox.focus ? 2 : 1
24 | border.color: "black"
25 | radius: 4
26 |
27 | Image {
28 | id: checkboxText
29 | source: "qrc:/imgs/images/tick.png"
30 | anchors.centerIn: parent
31 | width: parent.width - 4
32 | height: parent.height - 4
33 | visible: checked
34 | }
35 | }
36 |
37 | Text {
38 | text: checkbox.text
39 | font.pixelSize: 10
40 | y: 4
41 | }
42 | }
43 |
44 | MouseArea {
45 | anchors.fill: parent
46 | onClicked: checkbox.checked = !checkbox.checked
47 | }
48 |
49 | Keys.onSpacePressed: checkbox.checked = !checkbox.checked
50 | }//checkbox
51 |
--------------------------------------------------------------------------------
/WifiButton.qml:
--------------------------------------------------------------------------------
1 | import QtQuick 2.0
2 |
3 | Rectangle {
4 | id: wifiButton
5 | signal clicked
6 |
7 | radius: 4
8 | border.color: "#cccccc"
9 |
10 |
11 | property alias wifiTite: txt.text
12 | property alias textColor: txt.color
13 | textColor: "white"
14 |
15 | property alias collapseImgSrc: collapseImg.source
16 |
17 | property alias imgSrc: img.source
18 |
19 | y: 5
20 |
21 | //---------------------------------------------------------------
22 | // put it before all other controls below
23 | // so that they appear on top of it with higher z index
24 | //---------------------------------------------------------------
25 | Button {
26 | color: parent.color
27 | anchors.centerIn: wifiButton
28 | anchors.fill: wifiButton
29 | onClicked: wifiButton.clicked()
30 | }
31 |
32 |
33 | Image {
34 | id: img
35 | source: "qrc:/imgs/images/wifi4.png"
36 | x: 6
37 | y: 6
38 | width: 48
39 | height: 48
40 | }
41 |
42 | Text {
43 | id: txt
44 | anchors.centerIn: wifiButton
45 | }
46 |
47 | Image {
48 | id: collapseImg
49 | source: "qrc:/imgs/images/down-white.png"
50 | x: wifiButton.width - width - 14
51 | y: 16
52 | width: 40
53 | height: 24
54 | }
55 |
56 |
57 | }//wifiButton
58 |
--------------------------------------------------------------------------------
/BlApp.pro:
--------------------------------------------------------------------------------
1 | TEMPLATE = app
2 |
3 | QT += qml quick webview
4 | CONFIG += c++11
5 |
6 | SOURCES += main.cpp \
7 | qmlbridge.cpp \
8 | wifi/wifi_scan.c \
9 | wifimanager.cpp \
10 | wifi/wifiinfo.cpp \
11 | utils.cpp
12 |
13 | RESOURCES += qml.qrc \
14 | images.qrc
15 |
16 | #LIBS += -lmnl
17 |
18 | # Additional import path used to resolve QML modules in Qt Creator's code model
19 | QML_IMPORT_PATH =
20 |
21 | # Additional import path used to resolve QML modules just for Qt Quick Designer
22 | QML_DESIGNER_IMPORT_PATH =
23 |
24 | # The following define makes your compiler emit warnings if you use
25 | # any feature of Qt which as been marked deprecated (the exact warnings
26 | # depend on your compiler). Please consult the documentation of the
27 | # deprecated API in order to know how to port your code away from it.
28 | DEFINES += QT_DEPRECATED_WARNINGS
29 |
30 | # You can also make your code fail to compile if you use deprecated APIs.
31 | # In order to do so, uncomment the following line.
32 | # You can also select to disable deprecated APIs only up to a certain version of Qt.
33 | #DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0x060000 # disables all the APIs deprecated before Qt 6.0.0
34 |
35 | # Default rules for deployment.
36 | qnx: target.path = /tmp/$${TARGET}/bin
37 | else: unix:!android: target.path = /opt/$${TARGET}/bin
38 | !isEmpty(target.path): INSTALLS += target
39 |
40 | DISTFILES +=
41 |
42 | HEADERS += \
43 | qmlbridge.h \
44 | wifi/wifi_scan.h \
45 | wifimanager.h \
46 | wifi/wifiinfo.h \
47 | utils.h
48 |
--------------------------------------------------------------------------------
/WebDashboard.qml:
--------------------------------------------------------------------------------
1 | import QtQuick 2.7
2 | import QtQuick.Window 2.2
3 | import QtWebView 1.1
4 |
5 | Rectangle {
6 |
7 | id: root
8 | visible: true
9 | width: 800
10 | height: 480
11 |
12 | signal loadPage(string page)
13 |
14 | property int barHeight: 60
15 |
16 | Rectangle {
17 | id: topBar
18 | width: parent.width
19 | height: barHeight
20 | color: "#000000"
21 |
22 | Text {
23 | anchors.centerIn: parent
24 | text: qsTr("Dashboard")
25 | color: "#ffffff"
26 | }
27 |
28 | Button {
29 | id: homeButton
30 | width: 50
31 | height: 50
32 | imgSrc: "qrc:/imgs/images/home-small.png"
33 | //anchors.verticalCenter: parent
34 | x: 6
35 | y: 6
36 | color: parent.color
37 | onClicked: {
38 | root.loadPage("Home.qml");
39 | }
40 | }
41 |
42 | }//topBar
43 |
44 | WebView {
45 | id: webView
46 | anchors.top: topBar.bottom
47 | height: parent.height - topBar.height - bottomBar.height
48 | width: parent.width
49 | url: "https://freeboard.io/board/G6qZxn"
50 | }
51 |
52 | //------------------------------------------------------------
53 | // BOTTOM BAR
54 | //------------------------------------------------------------
55 | Rectangle {
56 | id: bottomBar
57 | anchors.top: webView.bottom
58 | width: parent.width
59 | height: barHeight
60 | color: "#000000"
61 | }//bottomBar
62 |
63 | }
64 |
--------------------------------------------------------------------------------
/wifimanager.cpp:
--------------------------------------------------------------------------------
1 | #include "wifimanager.h"
2 |
3 | #include "wifi/wifi_scan.h"
4 | #include "wifi/wifiinfo.h"
5 |
6 | #include
7 |
8 | #include
9 |
10 | #if 0
11 | //convert bssid to printable hardware mac address
12 | const char *bssid_to_string(const uint8_t bssid[BSSID_LENGTH], char bssid_string[BSSID_STRING_LENGTH]) {
13 | snprintf(bssid_string, BSSID_STRING_LENGTH, "%02x:%02x:%02x:%02x:%02x:%02x",
14 | bssid[0], bssid[1], bssid[2], bssid[3], bssid[4], bssid[5]);
15 | return bssid_string;
16 | }
17 |
18 | static const int BSS_INFOS=10; //the maximum amounts of APs (Access Points) we want to store
19 |
20 |
21 |
22 | QList WifiManager::getWifiNetworks() {
23 |
24 | dataList.clear();
25 |
26 | struct wifi_scan *wifi=NULL; //this stores all the library information
27 | struct bss_info bss[BSS_INFOS]; //this is where we are going to keep informatoin about APs (Access Points)
28 | char mac[BSSID_STRING_LENGTH]; //a placeholder where we convert BSSID to printable hardware mac address
29 | int status, i;
30 |
31 | printf("Scanning wifi networks...\n");
32 |
33 | // initialize the library with network interface argv[1] (e.g. wlan0)
34 | //wifi=wifi_scan_init("wlp3s0");
35 |
36 | wifi=wifi_scan_init("wlx00018edc151e");
37 |
38 | status=wifi_scan_all(wifi, bss, BSS_INFOS);
39 |
40 | //it may happen that device is unreachable (e.g. the device works in such way that it doesn't respond while scanning)
41 | //you may test for errno==EBUSY here and make a retry after a while, this is how my hardware works for example
42 | if(status<0) {
43 | perror("Unable to get scan data");
44 | }
45 | else {
46 | // wifi_scan_all returns the number of found stations,
47 | // it may be greater than BSS_INFOS that's why we test for both in the loop
48 | for(i=0;i WifiManager::getWifiNetworks() {
67 |
68 | dataList.clear();
69 |
70 | //QString str = "AIRTEL_E5172_B709:-53+Airtel-B310-0572:-40+JioFi2_9F03C0:-71+JioFi2_A885EA:-65";
71 |
72 | QString str = "AIRTEL_E5172_B709:-53+JioFi2_9F03C0:-71+Airtel-B310-0572:-40";
73 |
74 | QStringList cells = str.split("+");
75 | for(int i=0; i< cells.length(); i++) {
76 | QString cell = cells.at(i);
77 |
78 | QStringList items = cell.split(":");
79 |
80 | QString SSID = items.at(0);
81 | int Strength = items.at(1).toInt();
82 |
83 | qDebug () << "SSID: " << SSID;
84 | qDebug () << "Strength: " << Strength;
85 |
86 | dataList.append(new WifiInfo(SSID, Strength));
87 | }
88 |
89 |
90 |
91 |
92 | return dataList;
93 | }
94 |
95 | void WifiManager::connectToWifi(QString name, QString pass) {
96 | qDebug () << "[WifiManager] Connecting to " << name << " : " << pass;
97 | }
98 |
--------------------------------------------------------------------------------
/wifi/wifi_scan.h:
--------------------------------------------------------------------------------
1 | /*
2 | * wifi-scan library header
3 | *
4 | * Copyright (C) 2016 Bartosz Meglicki
5 | *
6 | * This program is free software; you can redistribute it and/or modify
7 | * it under the terms of the GNU General Public License version 3 as
8 | * published by the Free Software Foundation.
9 | * This program is distributed "as is" WITHOUT ANY WARRANTY of any
10 | * kind, whether express or implied; without even the implied warranty
11 | * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 | * GNU General Public License for more details.
13 | */
14 |
15 | #if 0
16 |
17 | #pragma once
18 |
19 | #ifdef __cplusplus
20 | extern "C" {
21 | #endif
22 |
23 | #include
24 |
25 | // some constants - mac address length, mac adress string length, max length of wireless network id with null character
26 | enum wifi_constants {BSSID_LENGTH=6, BSSID_STRING_LENGTH=18, SSID_MAX_LENGTH_WITH_NULL=33};
27 | // anything >=0 should mean that your are associated with the station
28 | enum bss_status{BSS_NONE=-1, BSS_AUTHENTHICATED=0, BSS_ASSOCIATED=1, BSS_IBSS_JOINED=2};
29 |
30 | // internal data used by the functions
31 | struct wifi_scan;
32 |
33 | // a single wireless network can have multiple BSSes working as network under one SSID
34 | struct bss_info
35 | {
36 | uint8_t bssid[BSSID_LENGTH]; //this is hardware mac address of your AP
37 | char ssid[SSID_MAX_LENGTH_WITH_NULL]; //this is the name of your AP as you see it when connecting
38 | enum bss_status status; //anything >=0 means that your are connected to this station/network
39 | int32_t signal_mbm; //signal strength in mBm, divide it by 100 to get signal in dBm
40 | int32_t seen_ms_ago; //when the above information was collected
41 | };
42 |
43 | // like above
44 | struct station_info
45 | {
46 | uint8_t bssid[BSSID_LENGTH]; //this is hardware mac address of your AP
47 | char ssid[SSID_MAX_LENGTH_WITH_NULL]; //this is the name of your AP as you see it when connecting
48 | enum bss_status status; //anything >=0 means that your are connected to this station/network
49 | int8_t signal_dbm; //signal strength in dBm from last received PPDU, you may need to average that
50 | uint32_t rx_packets; //the number of received packets
51 | uint32_t tx_packets; //the number of transmitted packets
52 | };
53 |
54 | /* Initializes the library
55 | *
56 | * If this functions fails the library will die with error message explaining why
57 | *
58 | * parameters:
59 | * interface - wireless interface, e.g. wlan0, wlan1
60 | *
61 | * returns:
62 | * struct wifi_scan * - pass it to all the functions in the library
63 | *
64 | */
65 | struct wifi_scan *wifi_scan_init(const char *interface);
66 |
67 | /* Frees the resources used by library
68 | *
69 | * parameters:
70 | * wifi - library data initialized with wifi_scan_init
71 | *
72 | * preconditions:
73 | * wifi initialized with wifi_scan_init
74 | */
75 | void wifi_scan_close(struct wifi_scan *wifi);
76 |
77 | /* Retrieve information about station you are associated to
78 | *
79 | * Retrieves information only about single station.
80 | * This function can be called repeateadly fast.
81 | *
82 | * parameters:
83 | * wifi - library data initialized with wifi_scan_init
84 | * station - to be filled with information
85 | *
86 | * returns:
87 | * -1 on error (errno is set), 0 if not associated to any station, 1 if data was retrieved
88 | *
89 | * preconditions:
90 | * wifi initialized with wifi_scan_init
91 | *
92 | */
93 | int wifi_scan_station(struct wifi_scan *wifi, struct station_info *station);
94 |
95 | /* Make a passive scan of all networks around.
96 | *
97 | * This function triggers passive scan if necessery, waits for completion and returns the data.
98 | * If some other scan was triggered in the meanwhile the library will collect it's results.
99 | * Triggering a scan requires permissions, for testing you may use sudo.
100 | *
101 | * Scanning may take some time (it can be order of second).
102 | * While scanning the link may be unusable for other programs!
103 | *
104 | * parameters:
105 | * wifi - library data initialized with wifi_scan_init
106 | * bss_infos - array of bss_info of size bss_infos_length
107 | * bss_infos_length - the length of passed array
108 | *
109 | * returns:
110 | * -1 on error (errno is set) or the number of found BSSes, the number may be greater then bss_infos_length
111 | *
112 | * Some devices may fail with -1 and errno=EBUSY if triggering scan when another scan is in progress. You may wait and retry in that case
113 | *
114 | * preconditions:
115 | * wifi initialized with wifi_scan_init
116 | *
117 | */
118 | int wifi_scan_all(struct wifi_scan *wifi, struct bss_info *bss_infos, int bss_infos_length);
119 |
120 | #ifdef __cplusplus
121 | }
122 | #endif
123 |
124 | #endif
125 |
--------------------------------------------------------------------------------
/Home.qml:
--------------------------------------------------------------------------------
1 | import QtQuick 2.7
2 | import QtQuick.Window 2.2
3 |
4 | Rectangle {
5 |
6 | id: root
7 | visible: true
8 | width: 800
9 | height: 480
10 |
11 | signal loadPage(string page)
12 |
13 | property int barHeight: 60
14 |
15 | function showPlaceholderPopup() {
16 | rectPlacehoder.visible = true;
17 | timerPlaceholder.start();
18 |
19 | }
20 |
21 | //------------------------------------------------------------
22 | // TOP BAR
23 | //------------------------------------------------------------
24 | Rectangle {
25 | id: topBar
26 | width: parent.width
27 | height: barHeight
28 | //color: "#1E323D"
29 | color: "#000000"
30 |
31 | Rectangle {
32 |
33 | width: topBar.width/2
34 | height: topBar.height
35 | color: topBar.color
36 | x: 20
37 | Text {
38 | anchors.centerIn: parent
39 | text: qsTr("My Device Name: " + qmlBridge.getDeivceName());
40 | color: "#ffffff"
41 | x: 60
42 | y: topBar.height + 20
43 | }
44 | }
45 |
46 |
47 | NotifyButton {
48 | id: btnNotify
49 | imgSrc: "qrc:/imgs/images/bell-gray.png"
50 | height: topBar.height - 8
51 | y: 4
52 | width: 100
53 | x: root.width * 6/10
54 | count: 90
55 |
56 | }
57 |
58 | Rectangle {
59 |
60 | id: wifiRect
61 | height: topBar.height
62 | width: 60
63 | anchors.left: btnNotify.right
64 | color: topBar.color
65 | Image {
66 | id: wifiIcon
67 | source: "qrc:/imgs/images/wifi4.png"
68 | anchors.centerIn: parent
69 |
70 | }
71 | }
72 |
73 | TimeRect {
74 | textColor: "#ffffff"
75 | height: topBar.height
76 | width: 100
77 | anchors.left: wifiRect.right
78 | color: topBar.color
79 | }
80 |
81 |
82 | }//topBar
83 |
84 | //------------------------------------------------------------
85 | // LEFT BUTTON
86 | //------------------------------------------------------------
87 | Button {
88 | id: leftButton
89 | width: barHeight - 4
90 | height: parent.height - topBar.height - bottomBar.height
91 | anchors.top: topBar.bottom
92 | imgSrc: "qrc:/imgs/images/left-arrow-gray.png"
93 | color: "#ffffff"
94 | imgWidth: 48
95 | imgHeight: 48
96 |
97 | onClicked: {
98 | console.log("Left button clicked");
99 | qmlBridge.test();
100 | qmlBridge.postMessage("hello from qml")
101 |
102 | console.log("wifis: " + qmlBridge.getWifiNetworks());
103 | var wifis = qmlBridge.getWifiNetworks();
104 | console.log(qmlBridge.getWifisCount());
105 |
106 | for(var i=0; i
5 | *
6 | * This program is free software; you can redistribute it and/or modify
7 | * it under the terms of the GNU General Public License version 3 as
8 | * published by the Free Software Foundation.
9 | * This program is distributed "as is" WITHOUT ANY WARRANTY of any
10 | * kind, whether express or implied; without even the implied warranty
11 | * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 | * GNU General Public License for more details.
13 | */
14 |
15 | #if 0
16 |
17 | /*
18 | * Library Overview
19 | *
20 | * This library uses netlink nl80211 user-space interface to retrieve wireless device information from kernel-space.
21 | * For netlink communication libmnl is used (minimalistic user-space netlink library).
22 | *
23 | * First concept you need to understand is that netlink uses sockets to communicate between user-space and kernel-space.
24 | *
25 | * There are 2 netlink communication channels (sockets/buffers)
26 | * - for notifications (about triggers, ready scan results)
27 | * - for commands (commanding triggers, retrieving scan results, station information)
28 | *
29 | * wifi_scan_init initializes 2 channels, gets nl80211 id using generic netlink (genetlink), gets id of
30 | * multicast group scan and subscribes to this group notifcations using notifications channel.
31 | *
32 | * wifi_scan_station gets the last (not necessarilly fresh) scan results that are available from the device,
33 | * checks which station we are associated with and retrieves information about this station (using commands channel)
34 | *
35 | * wifi_scan_all reads up any pending notifications, commands a trigger if necessary, waits for the device to gather
36 | * results and finally reads scan results with get_scan function (those are fresh results)
37 | *
38 | * wifi_scan_close frees up resources of two channels and any other resoureces that library uses.
39 | *
40 | * prepare_nl_messsage/send_nl_message/receive_nl_message are helper functions to simplify common tasks when issuing commands
41 | *
42 | * validate function simplifies common tasks (validates each attribute against table specifying what is valid)
43 | *
44 | */
45 |
46 | #include "wifi_scan.h"
47 |
48 | #include //netlink libmnl
49 | #include //nl80211 netlink
50 | #include //generic netlink
51 |
52 | #include
53 | #include
54 | #include
55 | #include
56 | #include
57 | #include //fntnl (set descriptor options)
58 | #include //errno
59 |
60 | // everything needed for sending/receiving with netlink
61 | struct netlink_channel
62 | {
63 | struct mnl_socket *nl; //netlink socket
64 | char *buf; //buffer for messages (in and out)
65 | uint16_t nl80211_id; //generic netlink nl80211 id
66 | uint32_t ifindex; //the wireless interface number (e.g. interface number for wlan0)
67 | uint32_t sequence; //the sequence number of netlink message
68 | void *context; //additional data to be stored/used when processing concrete message
69 | };
70 |
71 | // internal library data passed around by user
72 | struct wifi_scan
73 | {
74 | struct netlink_channel notification_channel;
75 | struct netlink_channel command_channel;
76 | };
77 |
78 | // DECLARATIONS AND TOP-DOWN LIBRARY OVERVIEW
79 |
80 | // INITIALIZATION
81 |
82 | // data needed from CTRL_CMD_GETFAMILY for nl80211, nl80211 id is stored in the channel rather then here
83 | struct context_CTRL_CMD_GETFAMILY
84 | {
85 | uint32_t id_NL80211_MULTICAST_GROUP_SCAN; //the id of group scan which we need to subscribe to
86 | };
87 |
88 | // public interface - initialize the library for wireless interface (e.g. wlan0)
89 | struct wifi_scan *wifi_scan_init(const char *interface);
90 |
91 | // allocate memory, set initial values, etc.
92 | void init_netlink_channel(struct netlink_channel *channel, const char *interface);
93 | // create netlink sockets for generic netlink
94 | void init_netlink_socket(struct netlink_channel *channel);
95 |
96 | // execute command to get nl80211 family and process the results
97 | int get_family_and_scan_ids(struct netlink_channel *channel);
98 | // this processes kernel reply for get family request, stores family id
99 | int handle_CTRL_CMD_GETFAMILY(const struct nlmsghdr *nlh, void *data);
100 | // parses multicast groups to get scan multicast group id
101 | void parse_CTRL_ATTR_MCAST_GROUPS(struct nlattr *nested, struct netlink_channel *channel);
102 |
103 | // subscribes channel to multicast group scan using scan group id
104 | void subscribe_NL80211_MULTICAST_GROUP_SCAN(struct netlink_channel *channel, uint32_t scan_group_id);
105 |
106 | // CLEANUP
107 |
108 | // public interface - cleans up after library
109 | void wifi_scan_close(struct wifi_scan *wifi);
110 | // cleans up after single channel
111 | void close_netlink_channel(struct netlink_channel *channel);
112 |
113 | // SCANNING
114 |
115 | // public interface - trigger scan if necessary, retrieve information about all known BSSes
116 | int wifi_scan_all(struct wifi_scan *wifi, struct bss_info *bss_infos, int bss_infos_length);
117 |
118 | // SCANNING - notification related
119 |
120 | // the data needed from notifications
121 | struct context_NL80211_MULTICAST_GROUP_SCAN
122 | {
123 | int new_scan_results; //are new scan results waiting for us?
124 | int scan_triggered; //was scan was already triggered by somebody else?
125 | };
126 |
127 | // read but do not block
128 | void read_past_notifications(struct netlink_channel *notifications);
129 | // go non-blocking
130 | void set_channel_non_blocking(struct netlink_channel *channel);
131 | // go back blocking
132 | void set_channel_blocking(struct netlink_channel *channel);
133 | // this handles notifications
134 | int handle_NL80211_MULTICAST_GROUP_SCAN(const struct nlmsghdr *nlh, void *data);
135 | // triggers scan if no results are waiting yet and if it was not already triggered
136 | int trigger_scan_if_necessary(struct netlink_channel *commands, struct context_NL80211_MULTICAST_GROUP_SCAN *scanning);
137 | // triggers the scan
138 | int trigger_scan(struct netlink_channel *channel);
139 | // wait for the notification that scan finished
140 | void wait_for_new_scan_results(struct netlink_channel *notifications);
141 |
142 | // SCANNING - scan related
143 |
144 | // the data needed from new scan results
145 | struct context_NL80211_CMD_NEW_SCAN_RESULTS
146 | {
147 | struct bss_info *bss_infos;
148 | int bss_infos_length;
149 | int scanned;
150 | };
151 |
152 | // get scan results cached by the driver
153 | int get_scan(struct netlink_channel *channel);
154 | // process the new scan results
155 | int handle_NL80211_CMD_NEW_SCAN_RESULTS(const struct nlmsghdr *nlh, void *data);
156 | // get the information about bss (nested attribute)
157 | void parse_NL80211_ATTR_BSS(struct nlattr *nested, struct netlink_channel *channel);
158 | // get the information from IE (non-netlink binary data here!)
159 | void parse_NL80211_BSS_INFORMATION_ELEMENTS(struct nlattr *attr, char SSID_OUT[33]);
160 | // get BSSID (mac address)
161 | void parse_NL80211_BSS_BSSID(struct nlattr *attr, uint8_t bssid_out[BSSID_LENGTH]);
162 |
163 | // STATION
164 |
165 | // data needed from command new station
166 | struct context_NL80211_CMD_NEW_STATION
167 | {
168 | struct station_info *station;
169 | };
170 |
171 | // public interface - get information about station we are associated with
172 | int wifi_scan_station(struct wifi_scan *wifi,struct station_info *station);
173 | // get information about station with BSSID
174 | int get_station(struct netlink_channel *channel, uint8_t bssid[BSSID_LENGTH]);
175 | // process command new station
176 | int handle_NL80211_CMD_NEW_STATION(const struct nlmsghdr *nlh, void *data);
177 | // process station info (nested attribute)
178 | void parse_NL80211_ATTR_STA_INFO(struct nlattr *nested, struct netlink_channel *channel);
179 |
180 | // NETLINK HELPERS
181 |
182 | // NETLINK HELPERS - message construction/sending/receiving
183 |
184 | // create the message with specified parameters for the channel
185 | // fill the message with additional attributes as needed with:
186 | // mnl_attr_put_[|u8|u16|u32|u64|str|strz] and mnl_attr_nest_[start|end]
187 | struct nlmsghdr *prepare_nl_message(uint32_t type, uint16_t flags, uint8_t genl_cmd, struct netlink_channel *channel);
188 | // send the above message
189 | void send_nl_message(struct nlmsghdr *nlh, struct netlink_channel *channel);
190 | // receive the results and process them using callback function
191 | int receive_nl_message(struct netlink_channel *channel, mnl_cb_t callback);
192 |
193 | // NETLINK HELPERS - validation
194 |
195 | // formal requirements for attribute
196 | struct attribute_validation
197 | {
198 | int attr; // attribute constant from nl80211.h
199 | enum mnl_attr_data_type type; // MNL_TYPE_[UNSPEC|U8|U16|U32|U64|STRING|FLAG|MSECS|NESTED|NESTED_COMPAT|NUL_STRING|BINARY]
200 | size_t len; // length in bytes, can be ommitted for attibutes of known size (e.g. U16), can be 0 if unspeciffied
201 | };
202 |
203 | // all information needed to validate attributes
204 | struct validation_data
205 | {
206 | struct nlattr **attribute_table; //validated attributes are returned here
207 | int attribute_length; //at most that many, distinct constants from nl80211.h go here
208 | const struct attribute_validation *validation; //vavildate against that table
209 | int validation_length;
210 | };
211 |
212 | // data of type struct validation_data*, validate attr against data, this is called for each attribute
213 | int validate(const struct nlattr *attr, void *data);
214 |
215 | // GENNERAL PURPOSE
216 |
217 | // if anything goes wrong...
218 | void die(const char *s);
219 | // as above but scream errno
220 | void die_errno(const char *s);
221 |
222 | // #####################################################################
223 | // IMPLEMENTATION
224 |
225 | // validate only what we are going to use, note that
226 | // this lists all the attributes used by the library
227 |
228 | const struct attribute_validation NL80211_VALIDATION[]={
229 | {CTRL_ATTR_FAMILY_ID, MNL_TYPE_U16},
230 | {CTRL_ATTR_MCAST_GROUPS, MNL_TYPE_NESTED} };
231 |
232 | const struct attribute_validation NL80211_MCAST_GROUPS_VALIDATION[]={
233 | {CTRL_ATTR_MCAST_GRP_ID, MNL_TYPE_U32},
234 | {CTRL_ATTR_MCAST_GRP_NAME, MNL_TYPE_STRING} };
235 |
236 | const struct attribute_validation NL80211_BSS_VALIDATION[]={
237 | {NL80211_BSS_BSSID, MNL_TYPE_BINARY, 6},
238 | {NL80211_BSS_INFORMATION_ELEMENTS, MNL_TYPE_BINARY},
239 | {NL80211_BSS_STATUS, MNL_TYPE_U32},
240 | {NL80211_BSS_SIGNAL_MBM, MNL_TYPE_U32},
241 | {NL80211_BSS_SEEN_MS_AGO, MNL_TYPE_U32} };
242 |
243 | const struct attribute_validation NL80211_NEW_SCAN_RESULTS_VALIDATION[]={
244 | {NL80211_ATTR_IFINDEX, MNL_TYPE_U32},
245 | {NL80211_ATTR_SCAN_SSIDS, MNL_TYPE_NESTED},
246 | {NL80211_ATTR_BSS, MNL_TYPE_NESTED} };
247 |
248 | const struct attribute_validation NL80211_CMD_NEW_STATION_VALIDATION[]={
249 | {NL80211_ATTR_STA_INFO, MNL_TYPE_NESTED},
250 | };
251 |
252 | const struct attribute_validation NL80211_STA_INFO_VALIDATION[]={
253 | {NL80211_STA_INFO_SIGNAL, MNL_TYPE_U8},
254 | {NL80211_STA_INFO_RX_PACKETS, MNL_TYPE_U32},
255 | {NL80211_STA_INFO_TX_PACKETS, MNL_TYPE_U32}
256 | };
257 |
258 | const int NL80211_VALIDATION_LENGTH=sizeof(NL80211_VALIDATION)/sizeof(struct attribute_validation);
259 | const int NL80211_MCAST_GROUPS_VALIDATION_LENGTH=sizeof(NL80211_MCAST_GROUPS_VALIDATION)/sizeof(struct attribute_validation);
260 | const int NL80211_BSS_VALIDATION_LENGTH=sizeof(NL80211_BSS_VALIDATION)/sizeof(struct attribute_validation);
261 | const int NL80211_NEW_SCAN_RESULTS_VALIDATION_LENGTH=sizeof(NL80211_NEW_SCAN_RESULTS_VALIDATION)/sizeof(struct attribute_validation);
262 | const int NL80211_CMD_NEW_STATION_VALIDATION_LENGTH=sizeof(NL80211_CMD_NEW_STATION_VALIDATION)/sizeof(struct attribute_validation);
263 | const int NL80211_STA_INFO_VALIDATION_LENGTH=sizeof(NL80211_STA_INFO_VALIDATION)/sizeof(struct attribute_validation);
264 |
265 | // INITIALIZATION
266 |
267 | // public interface - pass wireless interface like wlan0
268 | struct wifi_scan *wifi_scan_init(const char *interface)
269 | {
270 | struct wifi_scan *wifi=malloc(sizeof(struct wifi_scan));
271 | if(wifi==NULL)
272 | die("Insufficient memory - malloc(sizeof(struct wifi_data)");
273 |
274 | init_netlink_channel(&wifi->notification_channel, interface);
275 |
276 | struct context_CTRL_CMD_GETFAMILY family_context ={0};
277 | wifi->notification_channel.context=&family_context;
278 |
279 | if(get_family_and_scan_ids(&wifi->notification_channel) == -1)
280 | die_errno("GetFamilyAndScanId failed");
281 |
282 | if(family_context.id_NL80211_MULTICAST_GROUP_SCAN == 0)
283 | die("No scan multicast group in generic netlink nl80211\n");
284 |
285 | init_netlink_channel(&wifi->command_channel, interface);
286 | wifi->command_channel.nl80211_id = wifi->notification_channel.nl80211_id;
287 |
288 | subscribe_NL80211_MULTICAST_GROUP_SCAN(&wifi->notification_channel, family_context.id_NL80211_MULTICAST_GROUP_SCAN);
289 |
290 | return wifi;
291 | }
292 |
293 | // prerequisities:
294 | // - proper interface, e.g. wlan0, wlan1
295 | void init_netlink_channel(struct netlink_channel *channel, const char *interface)
296 | {
297 | channel->sequence=1;
298 | channel->buf=(char*) malloc(MNL_SOCKET_BUFFER_SIZE);
299 |
300 | if(channel->buf == NULL)
301 | die("Insufficent memory for netlink socket buffer");
302 |
303 | channel->ifindex=if_nametoindex(interface);
304 |
305 | if(channel->ifindex==0)
306 | die_errno("Incorrect network interface");
307 |
308 | channel->context=NULL;
309 |
310 | init_netlink_socket(channel);
311 | }
312 |
313 | void init_netlink_socket(struct netlink_channel *channel)
314 | {
315 | channel->nl = mnl_socket_open(NETLINK_GENERIC);
316 |
317 | if (channel->nl == NULL)
318 | die_errno("mnl_socket_open");
319 |
320 | if (mnl_socket_bind(channel->nl, 0, MNL_SOCKET_AUTOPID) < 0)
321 | die_errno("mnl_socket_bind");
322 | }
323 |
324 | // prerequisities:
325 | // - channel initialized with init_netlink_channel
326 | // - channel context of type context_CTRL_CMD_GETFAMILY
327 | int get_family_and_scan_ids(struct netlink_channel *channel)
328 | {
329 | struct nlmsghdr *nlh=prepare_nl_message(GENL_ID_CTRL, NLM_F_REQUEST | NLM_F_ACK, CTRL_CMD_GETFAMILY, channel);
330 | mnl_attr_put_u32(nlh, CTRL_ATTR_FAMILY_ID, GENL_ID_CTRL);
331 | mnl_attr_put_strz(nlh, CTRL_ATTR_FAMILY_NAME, NL80211_GENL_NAME);
332 |
333 | send_nl_message(nlh, channel);
334 |
335 | return receive_nl_message(channel, handle_CTRL_CMD_GETFAMILY);
336 | }
337 |
338 | // prerequisities:
339 | // - netlink_channel passed as data
340 | // - data->context of type struct context_CTRL_CMD_GETFAMILY
341 | int handle_CTRL_CMD_GETFAMILY(const struct nlmsghdr *nlh, void *data)
342 | {
343 | struct nlattr *tb[CTRL_ATTR_MAX+1] = {};
344 | struct genlmsghdr *genl = (struct genlmsghdr *)mnl_nlmsg_get_payload(nlh);
345 | struct netlink_channel *channel = (struct netlink_channel*)data;
346 | struct validation_data vd={tb, CTRL_ATTR_MAX, NL80211_VALIDATION, NL80211_VALIDATION_LENGTH};
347 |
348 | mnl_attr_parse(nlh, sizeof(*genl), validate, &vd);
349 |
350 | if (!tb[CTRL_ATTR_FAMILY_ID])
351 | die("No family id attribute");
352 |
353 | channel->nl80211_id=mnl_attr_get_u16(tb[CTRL_ATTR_FAMILY_ID]);
354 |
355 | if (tb[CTRL_ATTR_MCAST_GROUPS])
356 | parse_CTRL_ATTR_MCAST_GROUPS(tb[CTRL_ATTR_MCAST_GROUPS], channel);
357 |
358 | return MNL_CB_OK;
359 | }
360 |
361 | // prerequisities:
362 | // - data->context of type struct context_CTRL_CMD_GETFAMILY
363 | void parse_CTRL_ATTR_MCAST_GROUPS(struct nlattr *nested, struct netlink_channel *channel)
364 | {
365 | struct nlattr *pos;
366 |
367 | mnl_attr_for_each_nested(pos, nested)
368 | {
369 | struct nlattr *tb[CTRL_ATTR_MCAST_GRP_MAX+1] = {};
370 | struct validation_data vd={tb, CTRL_ATTR_MCAST_GRP_MAX, NL80211_MCAST_GROUPS_VALIDATION, NL80211_MCAST_GROUPS_VALIDATION_LENGTH};
371 |
372 | mnl_attr_parse_nested(pos, validate, &vd);
373 |
374 | if ( tb[CTRL_ATTR_MCAST_GRP_NAME])
375 | {
376 | const char *name=mnl_attr_get_str(tb[CTRL_ATTR_MCAST_GRP_NAME]);
377 |
378 | if( strcmp(name, "scan") == 0 )
379 | {
380 | if (tb[CTRL_ATTR_MCAST_GRP_ID])
381 | {
382 | struct context_CTRL_CMD_GETFAMILY *context=channel->context;
383 | context->id_NL80211_MULTICAST_GROUP_SCAN= mnl_attr_get_u32(tb[CTRL_ATTR_MCAST_GRP_ID]);
384 | }
385 | else
386 | die("Missing id attribute for scan multicast group");
387 | }
388 | }
389 | }
390 | }
391 |
392 | // prerequisities:
393 | // - channel initialized with init_netlink_channel
394 | void subscribe_NL80211_MULTICAST_GROUP_SCAN(struct netlink_channel *channel, uint32_t scan_group_id)
395 | {
396 | if (mnl_socket_setsockopt(channel->nl, NETLINK_ADD_MEMBERSHIP, &scan_group_id, sizeof(int)) < 0)
397 | die_errno("mnl_socket_set_sockopt");
398 | }
399 |
400 | // CLEANUP
401 |
402 | // prerequisities:
403 | // - wifi initialized with wifi_scan_init
404 | void wifi_scan_close(struct wifi_scan *wifi)
405 | {
406 | close_netlink_channel(&wifi->notification_channel);
407 | close_netlink_channel(&wifi->command_channel);
408 | free(wifi);
409 | }
410 |
411 | // prerequisities:
412 | // - channel initalized with init_netlink-channel
413 | void close_netlink_channel(struct netlink_channel *channel)
414 | {
415 | free(channel->buf);
416 | mnl_socket_close(channel->nl);
417 | }
418 |
419 |
420 | // SCANNING
421 |
422 | // handle also trigger abort
423 | // public interface
424 | //
425 | // prerequisities:
426 | // - wifi initialized with wifi_scan_init
427 | // - bss_info table of sized bss_info_length passed
428 | int wifi_scan_all(struct wifi_scan *wifi, struct bss_info *bss_infos, int bss_infos_length)
429 | {
430 | struct netlink_channel *notifications=&wifi->notification_channel;
431 | struct context_NL80211_MULTICAST_GROUP_SCAN scanning={0,0};
432 | notifications->context=&scanning;
433 |
434 | struct netlink_channel *commands=&wifi->command_channel;
435 | struct context_NL80211_CMD_NEW_SCAN_RESULTS scan_results = {bss_infos, bss_infos_length, 0};
436 | commands->context=&scan_results;
437 |
438 | //somebody else might have triggered scanning or even the results can be already waiting
439 | read_past_notifications(notifications);
440 |
441 | //if no results yet or scan not triggered then trigger it.
442 | //the device can be busy - we have to take it into account
443 | if( trigger_scan_if_necessary(commands, &scanning) == -1)
444 | return -1; //most likely with errno set to EBUSY
445 |
446 | //now just wait for trigger/new_scan_results
447 | wait_for_new_scan_results(notifications);
448 |
449 | //finally read the scan
450 | get_scan(commands);
451 |
452 | return scan_results.scanned;
453 | }
454 |
455 | // SCANNING - notification related
456 |
457 | // prerequisities
458 | // - subscribed to scan group with subscribe_NL80211_MULTICAST_GROUP_SCAN
459 | // - context_NL80211_MULTICAST_GROUP_SCAN set for notifications
460 | void read_past_notifications(struct netlink_channel *notifications)
461 | {
462 | set_channel_non_blocking(notifications);
463 | int ret, run_ret;
464 |
465 | while( (ret = mnl_socket_recvfrom(notifications->nl, notifications->buf, MNL_SOCKET_BUFFER_SIZE) ) >= 0)
466 | {
467 | //the line below fills context about past scans/triggers
468 | run_ret = mnl_cb_run(notifications->buf, ret, 0, 0, handle_NL80211_MULTICAST_GROUP_SCAN, notifications);
469 | if(run_ret <= 0)
470 | die_errno("ReadPastNotificationsNonBlocking mnl_cb_run failed");
471 | }
472 |
473 | if(ret == -1)
474 | if( !(errno == EINPROGRESS || errno == EWOULDBLOCK) )
475 | die_errno("ReadPastNotificationsNonBlocking mnl_socket_recv failed");
476 | //no more notifications waiting
477 | set_channel_blocking(notifications);
478 | }
479 |
480 | // prerequisities
481 | // - channel initialized with init_netlink_channel
482 | void set_channel_non_blocking(struct netlink_channel *channel)
483 | {
484 | int fd = mnl_socket_get_fd(channel->nl);
485 | int flags = fcntl(fd, F_GETFL, 0);
486 | if(flags == -1)
487 | die_errno("SetChannelNonBlocking F_GETFL");
488 | if( fcntl(fd, F_SETFL, flags | O_NONBLOCK) == -1)
489 | die_errno("SetChannelNonBlocking F_SETFL");
490 | }
491 |
492 | // prerequisities
493 | // - channel initialized with init_netlink_channel
494 | void set_channel_blocking(struct netlink_channel *channel)
495 | {
496 | int fd = mnl_socket_get_fd(channel->nl);
497 | int flags = fcntl(fd, F_GETFL, 0);
498 | if(flags == -1)
499 | die_errno("SetChannelNonBlocking F_GETFL");
500 | if( fcntl(fd, F_SETFL, flags & ~O_NONBLOCK) == -1)
501 | die_errno("SetChannelNonBlocking F_SETFL");
502 | }
503 |
504 | // prerequisities:
505 | // - subscribed to scan group with subscribe_NL80211_MULTICAST_GROUP_SCAN
506 | // - netlink_channel passed as data
507 | // - data->context of type struct context_NL80211_MULTICAST_GROUP_SCAN
508 | int handle_NL80211_MULTICAST_GROUP_SCAN(const struct nlmsghdr *nlh, void *data)
509 | {
510 | struct netlink_channel *channel=data;
511 | struct context_NL80211_MULTICAST_GROUP_SCAN *context = channel->context;
512 |
513 | struct genlmsghdr *genl = (struct genlmsghdr *)mnl_nlmsg_get_payload(nlh);
514 |
515 | // printf("Got message type %d seq %d pid %d genl cmd %d \n", nlh->nlmsg_type, nlh->nlmsg_seq, nlh->nlmsg_pid, genl->cmd);
516 | if(genl->cmd == NL80211_CMD_TRIGGER_SCAN)
517 | {
518 | context->scan_triggered=1;
519 | // printf("TRIGGER type %u seq %u pid %u genl cmd %u\n", nlh->nlmsg_type, nlh->nlmsg_seq, nlh->nlmsg_pid, genl->cmd);
520 | return MNL_CB_OK; //do nothing for now
521 | }
522 | else if(genl->cmd == NL80211_CMD_NEW_SCAN_RESULTS)
523 | {
524 | // printf("NEW SCAN RESULTS type %u seq %u pid %u genl cmd %u\n", nlh->nlmsg_type, nlh->nlmsg_seq, nlh->nlmsg_pid, genl->cmd);
525 | if(nlh->nlmsg_pid==0 && nlh->nlmsg_seq==0)
526 | context->new_scan_results = 1;
527 | return MNL_CB_OK; //do nothing for now
528 | }
529 | else
530 | {
531 | fprintf(stderr, "Ignoring generic netlink command type %u seq %u pid %u genl cmd %u\n",nlh->nlmsg_type, nlh->nlmsg_seq, nlh->nlmsg_pid, genl->cmd);
532 | return MNL_CB_OK;
533 | }
534 | }
535 |
536 |
537 | // prerequisities:
538 | // - commands initialized with init_netlink_channel
539 | // - scanning updated with read_past_notifications
540 | int trigger_scan_if_necessary(struct netlink_channel *commands, struct context_NL80211_MULTICAST_GROUP_SCAN *scanning)
541 | {
542 | if(!scanning->new_scan_results && !scanning->scan_triggered)
543 | if(trigger_scan(commands) == -1)
544 | return -1; //most likely errno set to EBUSY which means hardware is doing something else, try again later
545 | return 0;
546 | }
547 |
548 | // prerequisities:
549 | // - channel initialized with init_netlink_channel
550 | int trigger_scan(struct netlink_channel *channel)
551 | {
552 | struct nlmsghdr *nlh=prepare_nl_message(channel->nl80211_id, NLM_F_REQUEST | NLM_F_ACK, NL80211_CMD_TRIGGER_SCAN, channel);
553 | mnl_attr_put_u32(nlh, NL80211_ATTR_IFINDEX, channel->ifindex);
554 | send_nl_message(nlh, channel);
555 | return receive_nl_message(channel, handle_NL80211_CMD_NEW_SCAN_RESULTS);
556 | }
557 |
558 | // prerequisities
559 | // - channel initalized with init_netlink_channel
560 | // - subscribed to scan group with subscribe_NL80211_MULTICAST_GROUP_SCAN
561 | // - context_NL80211_MULTICAST_GROUP_SCAN set for notifications
562 | void wait_for_new_scan_results(struct netlink_channel *notifications)
563 | {
564 | struct context_NL80211_MULTICAST_GROUP_SCAN *scanning=notifications->context;
565 | int ret;
566 |
567 | while(!scanning->new_scan_results)
568 | {
569 | if ( (ret = mnl_socket_recvfrom(notifications->nl, notifications->buf, MNL_SOCKET_BUFFER_SIZE)) <=0 )
570 | die_errno("Waiting for new scan results failed - mnl_socket_recvfrom");
571 |
572 | if ( (ret=mnl_cb_run(notifications->buf, ret, 0, 0, handle_NL80211_MULTICAST_GROUP_SCAN, notifications)) <=0 )
573 | die_errno("Processing notificatoins failed - mnl_cb_run");
574 | }
575 | }
576 |
577 | // SCANNING - scan related
578 |
579 | // prerequisities:
580 | // - channel initalized with init_netlink_channel
581 | // - channel context of type context_NL80211_CMD_NEW_SCAN_RESULTS
582 | int get_scan(struct netlink_channel *channel)
583 | {
584 | struct nlmsghdr *nlh=prepare_nl_message(channel->nl80211_id, NLM_F_REQUEST | NLM_F_DUMP | NLM_F_ACK, NL80211_CMD_GET_SCAN, channel);
585 | mnl_attr_put_u32(nlh, NL80211_ATTR_IFINDEX, channel->ifindex);
586 |
587 | send_nl_message(nlh, channel);
588 | return receive_nl_message(channel, handle_NL80211_CMD_NEW_SCAN_RESULTS);
589 | }
590 |
591 | // prerequisities:
592 | // - netlink_channel passed as data
593 | // - data->context of type context_NL80211_CMD_NEW_SCAN_RESULTS
594 | int handle_NL80211_CMD_NEW_SCAN_RESULTS(const struct nlmsghdr *nlh, void *data)
595 | {
596 | struct netlink_channel *channel=data;
597 | struct nlattr *tb[NL80211_ATTR_MAX+1] = {};
598 | struct validation_data vd={tb, NL80211_ATTR_MAX, NL80211_NEW_SCAN_RESULTS_VALIDATION, NL80211_NEW_SCAN_RESULTS_VALIDATION_LENGTH};
599 | struct genlmsghdr *genl = (struct genlmsghdr *)mnl_nlmsg_get_payload(nlh);
600 |
601 | // printf("NSR type %u seq %u pid %u genl cmd %u\n", nlh->nlmsg_type, nlh->nlmsg_seq, nlh->nlmsg_pid, genl->cmd);
602 |
603 | if(genl->cmd != NL80211_CMD_NEW_SCAN_RESULTS)
604 | {
605 | fprintf(stderr, "Ignoring generic netlink command %u seq %u pid %u genl cmd %u\n", nlh->nlmsg_type, nlh->nlmsg_seq, nlh->nlmsg_pid, genl->cmd);
606 | return MNL_CB_OK;
607 | }
608 |
609 | //seq 0 - notification from kernel, then pid should also 0, if it is result of our scan we have sequence and our pid
610 | // int new_scan_results= nlh->nlmsg_seq != 0 && nlh->nlmsg_pid != 0;
611 |
612 | mnl_attr_parse(nlh, sizeof(*genl), validate, &vd);
613 |
614 | if(tb[NL80211_ATTR_IFINDEX])
615 | {
616 | // uint32_t ifindex=mnl_attr_get_u32(tb[NL80211_ATTR_IFINDEX]);
617 | // printf("ifindex %u\n", ifindex);
618 | }
619 | if (!tb[NL80211_ATTR_BSS])
620 | return MNL_CB_OK;
621 |
622 | parse_NL80211_ATTR_BSS(tb[NL80211_ATTR_BSS], channel);
623 |
624 | return MNL_CB_OK;
625 | }
626 |
627 | // prerequisities:
628 | // - channel context of type context_NL80211_CMD_NEW_SCAN_RESULTS
629 | void parse_NL80211_ATTR_BSS(struct nlattr *nested, struct netlink_channel *channel)
630 | {
631 | struct nlattr *tb[NL80211_BSS_MAX+1] = {};
632 | struct validation_data vd={tb, NL80211_BSS_MAX, NL80211_BSS_VALIDATION, NL80211_BSS_VALIDATION_LENGTH};
633 | struct context_NL80211_CMD_NEW_SCAN_RESULTS *scan_results = channel->context;
634 | struct bss_info *bss = scan_results->bss_infos + scan_results->scanned;
635 |
636 | mnl_attr_parse_nested(nested, validate, &vd);
637 |
638 | enum nl80211_bss_status status=BSS_NONE;
639 |
640 | if(tb[NL80211_BSS_STATUS])
641 | status=mnl_attr_get_u32(tb[NL80211_BSS_STATUS]);
642 |
643 | //if we have found associated station store first as last and associated as first
644 | if(status==NL80211_BSS_STATUS_ASSOCIATED || status==NL80211_BSS_STATUS_ASSOCIATED || status==NL80211_BSS_STATUS_IBSS_JOINED)
645 | {
646 | if(scan_results->scanned>0 && scan_results->scanned < scan_results->bss_infos_length)
647 | memcpy(bss, scan_results->bss_infos, sizeof(struct bss_info));
648 | bss=scan_results->bss_infos;
649 | }
650 |
651 | //check bounds, make exception if we have found associated station and replace previous data
652 | if(scan_results->bss_infos_length == 0 || ( scan_results->scanned >= scan_results->bss_infos_length && bss != scan_results->bss_infos ) )
653 | {
654 | ++scan_results->scanned;
655 | return;
656 | }
657 |
658 | if ( tb[NL80211_BSS_BSSID])
659 | parse_NL80211_BSS_BSSID(tb[NL80211_BSS_BSSID], bss->bssid);
660 |
661 | if ( tb[NL80211_BSS_INFORMATION_ELEMENTS])
662 | parse_NL80211_BSS_INFORMATION_ELEMENTS(tb[NL80211_BSS_INFORMATION_ELEMENTS], bss->ssid);
663 |
664 | if ( tb[NL80211_BSS_SIGNAL_MBM])
665 | bss->signal_mbm=mnl_attr_get_u32(tb[NL80211_BSS_SIGNAL_MBM]);
666 |
667 | if ( tb[NL80211_BSS_SEEN_MS_AGO])
668 | bss->seen_ms_ago = mnl_attr_get_u32(tb[NL80211_BSS_SEEN_MS_AGO]);
669 |
670 | bss->status=status;
671 |
672 | ++scan_results->scanned;
673 | }
674 |
675 | //This is guesswork! Read up on that!!! I don't think it's netlink in this attribute, some lower beacon layer
676 | void parse_NL80211_BSS_INFORMATION_ELEMENTS(struct nlattr *attr, char SSID_OUT[33])
677 | {
678 | const char *payload=mnl_attr_get_payload(attr);
679 | int len=mnl_attr_get_payload_len(attr);
680 | if(len==0 || payload[0]!=0 || payload[1] > 32 || payload[1] > len-2)
681 | {
682 | fprintf(stderr, "SSID len 0 or payload not starting from 0 or payload length > 32 or payload length > length-2\n!");
683 | SSID_OUT="\0";
684 | return;
685 | }
686 | int ssid_len=payload[1];
687 | strncpy(SSID_OUT, payload+2, ssid_len);
688 | SSID_OUT[ssid_len]='\0';
689 | }
690 |
691 | void parse_NL80211_BSS_BSSID(struct nlattr *attr, uint8_t bssid_out[BSSID_LENGTH])
692 | {
693 | const char *payload=mnl_attr_get_payload(attr);
694 | int len=mnl_attr_get_payload_len(attr);
695 |
696 | if(len != BSSID_LENGTH)
697 | {
698 | fprintf(stderr, "BSSID length != %d, ignoring", BSSID_LENGTH);
699 | memset(bssid_out, 0, BSSID_LENGTH);
700 | return;
701 | }
702 |
703 | memcpy(bssid_out, payload, BSSID_LENGTH);
704 | }
705 |
706 | // STATION
707 |
708 | // public interface
709 | //
710 | // prerequisities:
711 | // - wifi initialized with wifi_scan_init
712 | int wifi_scan_station(struct wifi_scan *wifi,struct station_info *station)
713 | {
714 | struct netlink_channel *commands=&wifi->command_channel;
715 | struct bss_info bss;
716 |
717 | struct context_NL80211_CMD_NEW_SCAN_RESULTS scan_results = {&bss, 1, 0};
718 | commands->context=&scan_results;
719 | get_scan(commands);
720 |
721 | if(scan_results.scanned==0)
722 | return 0;
723 |
724 | struct context_NL80211_CMD_NEW_STATION station_results = {station};
725 | commands->context=&station_results;
726 | get_station(commands, bss.bssid);
727 |
728 | memcpy(station->bssid, bss.bssid, BSSID_LENGTH);
729 | memcpy(station->ssid, bss.ssid, SSID_MAX_LENGTH_WITH_NULL);
730 | station->status=bss.status;
731 |
732 | return 1;
733 | }
734 |
735 | // prerequisites:
736 | // - channel initalized with init_netlink_channel
737 | // - context_NL80211_CMD_NEW_STATION set for channel
738 | int get_station(struct netlink_channel *channel, uint8_t bssid[BSSID_LENGTH])
739 | {
740 | struct nlmsghdr *nlh=prepare_nl_message(channel->nl80211_id, NLM_F_REQUEST | NLM_F_ACK, NL80211_CMD_GET_STATION, channel);
741 | mnl_attr_put_u32(nlh, NL80211_ATTR_IFINDEX, channel->ifindex);
742 | mnl_attr_put(nlh, NL80211_ATTR_MAC, BSSID_LENGTH, bssid);
743 | send_nl_message(nlh, channel);
744 | return receive_nl_message(channel, handle_NL80211_CMD_NEW_STATION);
745 | }
746 |
747 | // prerequisities:
748 | // - netlink_channel passed as data
749 | // - data->context of type context_NL80211_CMD_NEW_STATION
750 | int handle_NL80211_CMD_NEW_STATION(const struct nlmsghdr *nlh, void *data)
751 | {
752 | struct netlink_channel *channel=data;
753 | struct nlattr *tb[NL80211_ATTR_MAX+1] = {};
754 | struct validation_data vd={tb, NL80211_ATTR_MAX, NL80211_CMD_NEW_STATION_VALIDATION, NL80211_CMD_NEW_STATION_VALIDATION_LENGTH};
755 | struct genlmsghdr *genl = (struct genlmsghdr *)mnl_nlmsg_get_payload(nlh);
756 |
757 | if(genl->cmd != NL80211_CMD_NEW_STATION)
758 | {
759 | fprintf(stderr, "Ignoring generic netlink command %u seq %u pid %u genl cmd %u\n", nlh->nlmsg_type, nlh->nlmsg_seq, nlh->nlmsg_pid, genl->cmd);
760 | return MNL_CB_OK;
761 | }
762 |
763 | mnl_attr_parse(nlh, sizeof(*genl), validate, &vd);
764 |
765 | if(!tb[NL80211_ATTR_STA_INFO]) //or error, no statoin
766 | return MNL_CB_OK;
767 |
768 | parse_NL80211_ATTR_STA_INFO(tb[NL80211_ATTR_STA_INFO], channel);
769 |
770 | return MNL_CB_OK;
771 |
772 | }
773 |
774 | // prerequisities:
775 | // - channel context of type context_NL80211_CMD_NEW_STATION
776 | void parse_NL80211_ATTR_STA_INFO(struct nlattr *nested, struct netlink_channel *channel)
777 | {
778 | struct nlattr *tb[NL80211_STA_INFO_MAX+1] = {};
779 | struct validation_data vd={tb, NL80211_STA_INFO_MAX, NL80211_STA_INFO_VALIDATION, NL80211_STA_INFO_VALIDATION_LENGTH};
780 | struct context_NL80211_CMD_NEW_STATION *station_results = channel->context;
781 | struct station_info *station= station_results->station;
782 |
783 | mnl_attr_parse_nested(nested, validate, &vd);
784 |
785 | if ( tb[NL80211_STA_INFO_SIGNAL])
786 | station->signal_dbm=(int8_t)mnl_attr_get_u8(tb[NL80211_STA_INFO_SIGNAL]);
787 | if (tb[NL80211_STA_INFO_RX_PACKETS])
788 | station->rx_packets=mnl_attr_get_u32(tb[NL80211_STA_INFO_RX_PACKETS]);
789 | if (tb[NL80211_STA_INFO_TX_PACKETS])
790 | station->tx_packets=mnl_attr_get_u32(tb[NL80211_STA_INFO_TX_PACKETS]);
791 | }
792 |
793 |
794 | // NETLINK HELPERS
795 |
796 | // NETLINK HELPERS - message construction/sending/receiving
797 |
798 | // prerequisities:
799 | // - channel initialized with init_netlink_channel
800 | struct nlmsghdr *prepare_nl_message(uint32_t type, uint16_t flags, uint8_t genl_cmd, struct netlink_channel *channel)
801 | {
802 | struct nlmsghdr *nlh;
803 | struct genlmsghdr *genl;
804 |
805 | nlh = mnl_nlmsg_put_header(channel->buf);
806 | nlh->nlmsg_type = type;
807 | nlh->nlmsg_flags = flags;
808 | nlh->nlmsg_seq = channel->sequence;
809 |
810 | genl = (struct genlmsghdr*)mnl_nlmsg_put_extra_header(nlh, sizeof(struct genlmsghdr));
811 | genl->cmd = genl_cmd;
812 | genl->version = 1;
813 | return nlh;
814 | }
815 |
816 | // prerequisities:
817 | // - prepare_nl_message called first
818 | // - mnl_attr_put_xxx used if additional attributes needed
819 | void send_nl_message(struct nlmsghdr *nlh, struct netlink_channel *channel)
820 | {
821 | if (mnl_socket_sendto(channel->nl, nlh, nlh->nlmsg_len) < 0)
822 | die_errno("mnl_socket_sendto");
823 | }
824 |
825 | // prerequisities:
826 | // - send_nl_message called first
827 | // - prerequisities for callback matched
828 | int receive_nl_message(struct netlink_channel *channel, mnl_cb_t callback)
829 | {
830 | int ret;
831 | unsigned int portid = mnl_socket_get_portid(channel->nl);
832 |
833 | ret = mnl_socket_recvfrom(channel->nl, channel->buf, MNL_SOCKET_BUFFER_SIZE);
834 |
835 | while (ret > 0)
836 | {
837 | ret = mnl_cb_run(channel->buf, ret, channel->sequence, portid, callback, channel);
838 | if (ret <= 0)
839 | break;
840 | ret = mnl_socket_recvfrom(channel->nl, channel->buf, MNL_SOCKET_BUFFER_SIZE);
841 | }
842 |
843 | ++channel->sequence;
844 |
845 | return ret;
846 | }
847 |
848 | // NETLINK HELPERS - validation
849 |
850 | // prerequisities:
851 | // - data of type validation_data
852 | int validate(const struct nlattr *attr, void *data)
853 | {
854 | struct validation_data *vd=data;
855 | const struct nlattr **tb = (const struct nlattr**) vd->attribute_table;
856 | int type = mnl_attr_get_type(attr) ,i;
857 |
858 | // printf("%d\n", type);
859 |
860 | if (mnl_attr_type_valid(attr, vd->attribute_length) < 0)
861 | return MNL_CB_OK;
862 |
863 | for(i=0; i < vd->validation_length;++i)
864 | if(type == vd->validation[i].attr)
865 | {
866 | int len=vd->validation[i].len;
867 | if(len==0 && mnl_attr_validate(attr, vd->validation[i].type) < 0)
868 | {
869 | perror("mnl_attr_validate error");
870 | return MNL_CB_ERROR;
871 | }
872 | if(len != 0 && mnl_attr_validate2(attr, vd->validation[i].type, len) < 0)
873 | {
874 | perror("mnl_attr_validate error");
875 | return MNL_CB_ERROR;
876 | }
877 | }
878 |
879 | tb[type] = attr;
880 | return MNL_CB_OK;
881 | }
882 |
883 | // GENNERAL PURPOSE
884 |
885 | void die(const char *s)
886 | {
887 | fprintf(stderr, s);
888 | fprintf(stderr, "\n");
889 | exit(1);
890 | }
891 |
892 | void die_errno(const char *s)
893 | {
894 | perror(s);
895 | exit(1);
896 | }
897 |
898 | #endif
899 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | GNU GENERAL PUBLIC LICENSE
2 | Version 3, 29 June 2007
3 |
4 | Copyright (C) 2007 Free Software Foundation, Inc.
5 | Everyone is permitted to copy and distribute verbatim copies
6 | of this license document, but changing it is not allowed.
7 |
8 | Preamble
9 |
10 | The GNU General Public License is a free, copyleft license for
11 | software and other kinds of works.
12 |
13 | The licenses for most software and other practical works are designed
14 | to take away your freedom to share and change the works. By contrast,
15 | the GNU General Public License is intended to guarantee your freedom to
16 | share and change all versions of a program--to make sure it remains free
17 | software for all its users. We, the Free Software Foundation, use the
18 | GNU General Public License for most of our software; it applies also to
19 | any other work released this way by its authors. You can apply it to
20 | your programs, too.
21 |
22 | When we speak of free software, we are referring to freedom, not
23 | price. Our General Public Licenses are designed to make sure that you
24 | have the freedom to distribute copies of free software (and charge for
25 | them if you wish), that you receive source code or can get it if you
26 | want it, that you can change the software or use pieces of it in new
27 | free programs, and that you know you can do these things.
28 |
29 | To protect your rights, we need to prevent others from denying you
30 | these rights or asking you to surrender the rights. Therefore, you have
31 | certain responsibilities if you distribute copies of the software, or if
32 | you modify it: responsibilities to respect the freedom of others.
33 |
34 | For example, if you distribute copies of such a program, whether
35 | gratis or for a fee, you must pass on to the recipients the same
36 | freedoms that you received. You must make sure that they, too, receive
37 | or can get the source code. And you must show them these terms so they
38 | know their rights.
39 |
40 | Developers that use the GNU GPL protect your rights with two steps:
41 | (1) assert copyright on the software, and (2) offer you this License
42 | giving you legal permission to copy, distribute and/or modify it.
43 |
44 | For the developers' and authors' protection, the GPL clearly explains
45 | that there is no warranty for this free software. For both users' and
46 | authors' sake, the GPL requires that modified versions be marked as
47 | changed, so that their problems will not be attributed erroneously to
48 | authors of previous versions.
49 |
50 | Some devices are designed to deny users access to install or run
51 | modified versions of the software inside them, although the manufacturer
52 | can do so. This is fundamentally incompatible with the aim of
53 | protecting users' freedom to change the software. The systematic
54 | pattern of such abuse occurs in the area of products for individuals to
55 | use, which is precisely where it is most unacceptable. Therefore, we
56 | have designed this version of the GPL to prohibit the practice for those
57 | products. If such problems arise substantially in other domains, we
58 | stand ready to extend this provision to those domains in future versions
59 | of the GPL, as needed to protect the freedom of users.
60 |
61 | Finally, every program is threatened constantly by software patents.
62 | States should not allow patents to restrict development and use of
63 | software on general-purpose computers, but in those that do, we wish to
64 | avoid the special danger that patents applied to a free program could
65 | make it effectively proprietary. To prevent this, the GPL assures that
66 | patents cannot be used to render the program non-free.
67 |
68 | The precise terms and conditions for copying, distribution and
69 | modification follow.
70 |
71 | TERMS AND CONDITIONS
72 |
73 | 0. Definitions.
74 |
75 | "This License" refers to version 3 of the GNU General Public License.
76 |
77 | "Copyright" also means copyright-like laws that apply to other kinds of
78 | works, such as semiconductor masks.
79 |
80 | "The Program" refers to any copyrightable work licensed under this
81 | License. Each licensee is addressed as "you". "Licensees" and
82 | "recipients" may be individuals or organizations.
83 |
84 | To "modify" a work means to copy from or adapt all or part of the work
85 | in a fashion requiring copyright permission, other than the making of an
86 | exact copy. The resulting work is called a "modified version" of the
87 | earlier work or a work "based on" the earlier work.
88 |
89 | A "covered work" means either the unmodified Program or a work based
90 | on the Program.
91 |
92 | To "propagate" a work means to do anything with it that, without
93 | permission, would make you directly or secondarily liable for
94 | infringement under applicable copyright law, except executing it on a
95 | computer or modifying a private copy. Propagation includes copying,
96 | distribution (with or without modification), making available to the
97 | public, and in some countries other activities as well.
98 |
99 | To "convey" a work means any kind of propagation that enables other
100 | parties to make or receive copies. Mere interaction with a user through
101 | a computer network, with no transfer of a copy, is not conveying.
102 |
103 | An interactive user interface displays "Appropriate Legal Notices"
104 | to the extent that it includes a convenient and prominently visible
105 | feature that (1) displays an appropriate copyright notice, and (2)
106 | tells the user that there is no warranty for the work (except to the
107 | extent that warranties are provided), that licensees may convey the
108 | work under this License, and how to view a copy of this License. If
109 | the interface presents a list of user commands or options, such as a
110 | menu, a prominent item in the list meets this criterion.
111 |
112 | 1. Source Code.
113 |
114 | The "source code" for a work means the preferred form of the work
115 | for making modifications to it. "Object code" means any non-source
116 | form of a work.
117 |
118 | A "Standard Interface" means an interface that either is an official
119 | standard defined by a recognized standards body, or, in the case of
120 | interfaces specified for a particular programming language, one that
121 | is widely used among developers working in that language.
122 |
123 | The "System Libraries" of an executable work include anything, other
124 | than the work as a whole, that (a) is included in the normal form of
125 | packaging a Major Component, but which is not part of that Major
126 | Component, and (b) serves only to enable use of the work with that
127 | Major Component, or to implement a Standard Interface for which an
128 | implementation is available to the public in source code form. A
129 | "Major Component", in this context, means a major essential component
130 | (kernel, window system, and so on) of the specific operating system
131 | (if any) on which the executable work runs, or a compiler used to
132 | produce the work, or an object code interpreter used to run it.
133 |
134 | The "Corresponding Source" for a work in object code form means all
135 | the source code needed to generate, install, and (for an executable
136 | work) run the object code and to modify the work, including scripts to
137 | control those activities. However, it does not include the work's
138 | System Libraries, or general-purpose tools or generally available free
139 | programs which are used unmodified in performing those activities but
140 | which are not part of the work. For example, Corresponding Source
141 | includes interface definition files associated with source files for
142 | the work, and the source code for shared libraries and dynamically
143 | linked subprograms that the work is specifically designed to require,
144 | such as by intimate data communication or control flow between those
145 | subprograms and other parts of the work.
146 |
147 | The Corresponding Source need not include anything that users
148 | can regenerate automatically from other parts of the Corresponding
149 | Source.
150 |
151 | The Corresponding Source for a work in source code form is that
152 | same work.
153 |
154 | 2. Basic Permissions.
155 |
156 | All rights granted under this License are granted for the term of
157 | copyright on the Program, and are irrevocable provided the stated
158 | conditions are met. This License explicitly affirms your unlimited
159 | permission to run the unmodified Program. The output from running a
160 | covered work is covered by this License only if the output, given its
161 | content, constitutes a covered work. This License acknowledges your
162 | rights of fair use or other equivalent, as provided by copyright law.
163 |
164 | You may make, run and propagate covered works that you do not
165 | convey, without conditions so long as your license otherwise remains
166 | in force. You may convey covered works to others for the sole purpose
167 | of having them make modifications exclusively for you, or provide you
168 | with facilities for running those works, provided that you comply with
169 | the terms of this License in conveying all material for which you do
170 | not control copyright. Those thus making or running the covered works
171 | for you must do so exclusively on your behalf, under your direction
172 | and control, on terms that prohibit them from making any copies of
173 | your copyrighted material outside their relationship with you.
174 |
175 | Conveying under any other circumstances is permitted solely under
176 | the conditions stated below. Sublicensing is not allowed; section 10
177 | makes it unnecessary.
178 |
179 | 3. Protecting Users' Legal Rights From Anti-Circumvention Law.
180 |
181 | No covered work shall be deemed part of an effective technological
182 | measure under any applicable law fulfilling obligations under article
183 | 11 of the WIPO copyright treaty adopted on 20 December 1996, or
184 | similar laws prohibiting or restricting circumvention of such
185 | measures.
186 |
187 | When you convey a covered work, you waive any legal power to forbid
188 | circumvention of technological measures to the extent such circumvention
189 | is effected by exercising rights under this License with respect to
190 | the covered work, and you disclaim any intention to limit operation or
191 | modification of the work as a means of enforcing, against the work's
192 | users, your or third parties' legal rights to forbid circumvention of
193 | technological measures.
194 |
195 | 4. Conveying Verbatim Copies.
196 |
197 | You may convey verbatim copies of the Program's source code as you
198 | receive it, in any medium, provided that you conspicuously and
199 | appropriately publish on each copy an appropriate copyright notice;
200 | keep intact all notices stating that this License and any
201 | non-permissive terms added in accord with section 7 apply to the code;
202 | keep intact all notices of the absence of any warranty; and give all
203 | recipients a copy of this License along with the Program.
204 |
205 | You may charge any price or no price for each copy that you convey,
206 | and you may offer support or warranty protection for a fee.
207 |
208 | 5. Conveying Modified Source Versions.
209 |
210 | You may convey a work based on the Program, or the modifications to
211 | produce it from the Program, in the form of source code under the
212 | terms of section 4, provided that you also meet all of these conditions:
213 |
214 | a) The work must carry prominent notices stating that you modified
215 | it, and giving a relevant date.
216 |
217 | b) The work must carry prominent notices stating that it is
218 | released under this License and any conditions added under section
219 | 7. This requirement modifies the requirement in section 4 to
220 | "keep intact all notices".
221 |
222 | c) You must license the entire work, as a whole, under this
223 | License to anyone who comes into possession of a copy. This
224 | License will therefore apply, along with any applicable section 7
225 | additional terms, to the whole of the work, and all its parts,
226 | regardless of how they are packaged. This License gives no
227 | permission to license the work in any other way, but it does not
228 | invalidate such permission if you have separately received it.
229 |
230 | d) If the work has interactive user interfaces, each must display
231 | Appropriate Legal Notices; however, if the Program has interactive
232 | interfaces that do not display Appropriate Legal Notices, your
233 | work need not make them do so.
234 |
235 | A compilation of a covered work with other separate and independent
236 | works, which are not by their nature extensions of the covered work,
237 | and which are not combined with it such as to form a larger program,
238 | in or on a volume of a storage or distribution medium, is called an
239 | "aggregate" if the compilation and its resulting copyright are not
240 | used to limit the access or legal rights of the compilation's users
241 | beyond what the individual works permit. Inclusion of a covered work
242 | in an aggregate does not cause this License to apply to the other
243 | parts of the aggregate.
244 |
245 | 6. Conveying Non-Source Forms.
246 |
247 | You may convey a covered work in object code form under the terms
248 | of sections 4 and 5, provided that you also convey the
249 | machine-readable Corresponding Source under the terms of this License,
250 | in one of these ways:
251 |
252 | a) Convey the object code in, or embodied in, a physical product
253 | (including a physical distribution medium), accompanied by the
254 | Corresponding Source fixed on a durable physical medium
255 | customarily used for software interchange.
256 |
257 | b) Convey the object code in, or embodied in, a physical product
258 | (including a physical distribution medium), accompanied by a
259 | written offer, valid for at least three years and valid for as
260 | long as you offer spare parts or customer support for that product
261 | model, to give anyone who possesses the object code either (1) a
262 | copy of the Corresponding Source for all the software in the
263 | product that is covered by this License, on a durable physical
264 | medium customarily used for software interchange, for a price no
265 | more than your reasonable cost of physically performing this
266 | conveying of source, or (2) access to copy the
267 | Corresponding Source from a network server at no charge.
268 |
269 | c) Convey individual copies of the object code with a copy of the
270 | written offer to provide the Corresponding Source. This
271 | alternative is allowed only occasionally and noncommercially, and
272 | only if you received the object code with such an offer, in accord
273 | with subsection 6b.
274 |
275 | d) Convey the object code by offering access from a designated
276 | place (gratis or for a charge), and offer equivalent access to the
277 | Corresponding Source in the same way through the same place at no
278 | further charge. You need not require recipients to copy the
279 | Corresponding Source along with the object code. If the place to
280 | copy the object code is a network server, the Corresponding Source
281 | may be on a different server (operated by you or a third party)
282 | that supports equivalent copying facilities, provided you maintain
283 | clear directions next to the object code saying where to find the
284 | Corresponding Source. Regardless of what server hosts the
285 | Corresponding Source, you remain obligated to ensure that it is
286 | available for as long as needed to satisfy these requirements.
287 |
288 | e) Convey the object code using peer-to-peer transmission, provided
289 | you inform other peers where the object code and Corresponding
290 | Source of the work are being offered to the general public at no
291 | charge under subsection 6d.
292 |
293 | A separable portion of the object code, whose source code is excluded
294 | from the Corresponding Source as a System Library, need not be
295 | included in conveying the object code work.
296 |
297 | A "User Product" is either (1) a "consumer product", which means any
298 | tangible personal property which is normally used for personal, family,
299 | or household purposes, or (2) anything designed or sold for incorporation
300 | into a dwelling. In determining whether a product is a consumer product,
301 | doubtful cases shall be resolved in favor of coverage. For a particular
302 | product received by a particular user, "normally used" refers to a
303 | typical or common use of that class of product, regardless of the status
304 | of the particular user or of the way in which the particular user
305 | actually uses, or expects or is expected to use, the product. A product
306 | is a consumer product regardless of whether the product has substantial
307 | commercial, industrial or non-consumer uses, unless such uses represent
308 | the only significant mode of use of the product.
309 |
310 | "Installation Information" for a User Product means any methods,
311 | procedures, authorization keys, or other information required to install
312 | and execute modified versions of a covered work in that User Product from
313 | a modified version of its Corresponding Source. The information must
314 | suffice to ensure that the continued functioning of the modified object
315 | code is in no case prevented or interfered with solely because
316 | modification has been made.
317 |
318 | If you convey an object code work under this section in, or with, or
319 | specifically for use in, a User Product, and the conveying occurs as
320 | part of a transaction in which the right of possession and use of the
321 | User Product is transferred to the recipient in perpetuity or for a
322 | fixed term (regardless of how the transaction is characterized), the
323 | Corresponding Source conveyed under this section must be accompanied
324 | by the Installation Information. But this requirement does not apply
325 | if neither you nor any third party retains the ability to install
326 | modified object code on the User Product (for example, the work has
327 | been installed in ROM).
328 |
329 | The requirement to provide Installation Information does not include a
330 | requirement to continue to provide support service, warranty, or updates
331 | for a work that has been modified or installed by the recipient, or for
332 | the User Product in which it has been modified or installed. Access to a
333 | network may be denied when the modification itself materially and
334 | adversely affects the operation of the network or violates the rules and
335 | protocols for communication across the network.
336 |
337 | Corresponding Source conveyed, and Installation Information provided,
338 | in accord with this section must be in a format that is publicly
339 | documented (and with an implementation available to the public in
340 | source code form), and must require no special password or key for
341 | unpacking, reading or copying.
342 |
343 | 7. Additional Terms.
344 |
345 | "Additional permissions" are terms that supplement the terms of this
346 | License by making exceptions from one or more of its conditions.
347 | Additional permissions that are applicable to the entire Program shall
348 | be treated as though they were included in this License, to the extent
349 | that they are valid under applicable law. If additional permissions
350 | apply only to part of the Program, that part may be used separately
351 | under those permissions, but the entire Program remains governed by
352 | this License without regard to the additional permissions.
353 |
354 | When you convey a copy of a covered work, you may at your option
355 | remove any additional permissions from that copy, or from any part of
356 | it. (Additional permissions may be written to require their own
357 | removal in certain cases when you modify the work.) You may place
358 | additional permissions on material, added by you to a covered work,
359 | for which you have or can give appropriate copyright permission.
360 |
361 | Notwithstanding any other provision of this License, for material you
362 | add to a covered work, you may (if authorized by the copyright holders of
363 | that material) supplement the terms of this License with terms:
364 |
365 | a) Disclaiming warranty or limiting liability differently from the
366 | terms of sections 15 and 16 of this License; or
367 |
368 | b) Requiring preservation of specified reasonable legal notices or
369 | author attributions in that material or in the Appropriate Legal
370 | Notices displayed by works containing it; or
371 |
372 | c) Prohibiting misrepresentation of the origin of that material, or
373 | requiring that modified versions of such material be marked in
374 | reasonable ways as different from the original version; or
375 |
376 | d) Limiting the use for publicity purposes of names of licensors or
377 | authors of the material; or
378 |
379 | e) Declining to grant rights under trademark law for use of some
380 | trade names, trademarks, or service marks; or
381 |
382 | f) Requiring indemnification of licensors and authors of that
383 | material by anyone who conveys the material (or modified versions of
384 | it) with contractual assumptions of liability to the recipient, for
385 | any liability that these contractual assumptions directly impose on
386 | those licensors and authors.
387 |
388 | All other non-permissive additional terms are considered "further
389 | restrictions" within the meaning of section 10. If the Program as you
390 | received it, or any part of it, contains a notice stating that it is
391 | governed by this License along with a term that is a further
392 | restriction, you may remove that term. If a license document contains
393 | a further restriction but permits relicensing or conveying under this
394 | License, you may add to a covered work material governed by the terms
395 | of that license document, provided that the further restriction does
396 | not survive such relicensing or conveying.
397 |
398 | If you add terms to a covered work in accord with this section, you
399 | must place, in the relevant source files, a statement of the
400 | additional terms that apply to those files, or a notice indicating
401 | where to find the applicable terms.
402 |
403 | Additional terms, permissive or non-permissive, may be stated in the
404 | form of a separately written license, or stated as exceptions;
405 | the above requirements apply either way.
406 |
407 | 8. Termination.
408 |
409 | You may not propagate or modify a covered work except as expressly
410 | provided under this License. Any attempt otherwise to propagate or
411 | modify it is void, and will automatically terminate your rights under
412 | this License (including any patent licenses granted under the third
413 | paragraph of section 11).
414 |
415 | However, if you cease all violation of this License, then your
416 | license from a particular copyright holder is reinstated (a)
417 | provisionally, unless and until the copyright holder explicitly and
418 | finally terminates your license, and (b) permanently, if the copyright
419 | holder fails to notify you of the violation by some reasonable means
420 | prior to 60 days after the cessation.
421 |
422 | Moreover, your license from a particular copyright holder is
423 | reinstated permanently if the copyright holder notifies you of the
424 | violation by some reasonable means, this is the first time you have
425 | received notice of violation of this License (for any work) from that
426 | copyright holder, and you cure the violation prior to 30 days after
427 | your receipt of the notice.
428 |
429 | Termination of your rights under this section does not terminate the
430 | licenses of parties who have received copies or rights from you under
431 | this License. If your rights have been terminated and not permanently
432 | reinstated, you do not qualify to receive new licenses for the same
433 | material under section 10.
434 |
435 | 9. Acceptance Not Required for Having Copies.
436 |
437 | You are not required to accept this License in order to receive or
438 | run a copy of the Program. Ancillary propagation of a covered work
439 | occurring solely as a consequence of using peer-to-peer transmission
440 | to receive a copy likewise does not require acceptance. However,
441 | nothing other than this License grants you permission to propagate or
442 | modify any covered work. These actions infringe copyright if you do
443 | not accept this License. Therefore, by modifying or propagating a
444 | covered work, you indicate your acceptance of this License to do so.
445 |
446 | 10. Automatic Licensing of Downstream Recipients.
447 |
448 | Each time you convey a covered work, the recipient automatically
449 | receives a license from the original licensors, to run, modify and
450 | propagate that work, subject to this License. You are not responsible
451 | for enforcing compliance by third parties with this License.
452 |
453 | An "entity transaction" is a transaction transferring control of an
454 | organization, or substantially all assets of one, or subdividing an
455 | organization, or merging organizations. If propagation of a covered
456 | work results from an entity transaction, each party to that
457 | transaction who receives a copy of the work also receives whatever
458 | licenses to the work the party's predecessor in interest had or could
459 | give under the previous paragraph, plus a right to possession of the
460 | Corresponding Source of the work from the predecessor in interest, if
461 | the predecessor has it or can get it with reasonable efforts.
462 |
463 | You may not impose any further restrictions on the exercise of the
464 | rights granted or affirmed under this License. For example, you may
465 | not impose a license fee, royalty, or other charge for exercise of
466 | rights granted under this License, and you may not initiate litigation
467 | (including a cross-claim or counterclaim in a lawsuit) alleging that
468 | any patent claim is infringed by making, using, selling, offering for
469 | sale, or importing the Program or any portion of it.
470 |
471 | 11. Patents.
472 |
473 | A "contributor" is a copyright holder who authorizes use under this
474 | License of the Program or a work on which the Program is based. The
475 | work thus licensed is called the contributor's "contributor version".
476 |
477 | A contributor's "essential patent claims" are all patent claims
478 | owned or controlled by the contributor, whether already acquired or
479 | hereafter acquired, that would be infringed by some manner, permitted
480 | by this License, of making, using, or selling its contributor version,
481 | but do not include claims that would be infringed only as a
482 | consequence of further modification of the contributor version. For
483 | purposes of this definition, "control" includes the right to grant
484 | patent sublicenses in a manner consistent with the requirements of
485 | this License.
486 |
487 | Each contributor grants you a non-exclusive, worldwide, royalty-free
488 | patent license under the contributor's essential patent claims, to
489 | make, use, sell, offer for sale, import and otherwise run, modify and
490 | propagate the contents of its contributor version.
491 |
492 | In the following three paragraphs, a "patent license" is any express
493 | agreement or commitment, however denominated, not to enforce a patent
494 | (such as an express permission to practice a patent or covenant not to
495 | sue for patent infringement). To "grant" such a patent license to a
496 | party means to make such an agreement or commitment not to enforce a
497 | patent against the party.
498 |
499 | If you convey a covered work, knowingly relying on a patent license,
500 | and the Corresponding Source of the work is not available for anyone
501 | to copy, free of charge and under the terms of this License, through a
502 | publicly available network server or other readily accessible means,
503 | then you must either (1) cause the Corresponding Source to be so
504 | available, or (2) arrange to deprive yourself of the benefit of the
505 | patent license for this particular work, or (3) arrange, in a manner
506 | consistent with the requirements of this License, to extend the patent
507 | license to downstream recipients. "Knowingly relying" means you have
508 | actual knowledge that, but for the patent license, your conveying the
509 | covered work in a country, or your recipient's use of the covered work
510 | in a country, would infringe one or more identifiable patents in that
511 | country that you have reason to believe are valid.
512 |
513 | If, pursuant to or in connection with a single transaction or
514 | arrangement, you convey, or propagate by procuring conveyance of, a
515 | covered work, and grant a patent license to some of the parties
516 | receiving the covered work authorizing them to use, propagate, modify
517 | or convey a specific copy of the covered work, then the patent license
518 | you grant is automatically extended to all recipients of the covered
519 | work and works based on it.
520 |
521 | A patent license is "discriminatory" if it does not include within
522 | the scope of its coverage, prohibits the exercise of, or is
523 | conditioned on the non-exercise of one or more of the rights that are
524 | specifically granted under this License. You may not convey a covered
525 | work if you are a party to an arrangement with a third party that is
526 | in the business of distributing software, under which you make payment
527 | to the third party based on the extent of your activity of conveying
528 | the work, and under which the third party grants, to any of the
529 | parties who would receive the covered work from you, a discriminatory
530 | patent license (a) in connection with copies of the covered work
531 | conveyed by you (or copies made from those copies), or (b) primarily
532 | for and in connection with specific products or compilations that
533 | contain the covered work, unless you entered into that arrangement,
534 | or that patent license was granted, prior to 28 March 2007.
535 |
536 | Nothing in this License shall be construed as excluding or limiting
537 | any implied license or other defenses to infringement that may
538 | otherwise be available to you under applicable patent law.
539 |
540 | 12. No Surrender of Others' Freedom.
541 |
542 | If conditions are imposed on you (whether by court order, agreement or
543 | otherwise) that contradict the conditions of this License, they do not
544 | excuse you from the conditions of this License. If you cannot convey a
545 | covered work so as to satisfy simultaneously your obligations under this
546 | License and any other pertinent obligations, then as a consequence you may
547 | not convey it at all. For example, if you agree to terms that obligate you
548 | to collect a royalty for further conveying from those to whom you convey
549 | the Program, the only way you could satisfy both those terms and this
550 | License would be to refrain entirely from conveying the Program.
551 |
552 | 13. Use with the GNU Affero General Public License.
553 |
554 | Notwithstanding any other provision of this License, you have
555 | permission to link or combine any covered work with a work licensed
556 | under version 3 of the GNU Affero General Public License into a single
557 | combined work, and to convey the resulting work. The terms of this
558 | License will continue to apply to the part which is the covered work,
559 | but the special requirements of the GNU Affero General Public License,
560 | section 13, concerning interaction through a network will apply to the
561 | combination as such.
562 |
563 | 14. Revised Versions of this License.
564 |
565 | The Free Software Foundation may publish revised and/or new versions of
566 | the GNU General Public License from time to time. Such new versions will
567 | be similar in spirit to the present version, but may differ in detail to
568 | address new problems or concerns.
569 |
570 | Each version is given a distinguishing version number. If the
571 | Program specifies that a certain numbered version of the GNU General
572 | Public License "or any later version" applies to it, you have the
573 | option of following the terms and conditions either of that numbered
574 | version or of any later version published by the Free Software
575 | Foundation. If the Program does not specify a version number of the
576 | GNU General Public License, you may choose any version ever published
577 | by the Free Software Foundation.
578 |
579 | If the Program specifies that a proxy can decide which future
580 | versions of the GNU General Public License can be used, that proxy's
581 | public statement of acceptance of a version permanently authorizes you
582 | to choose that version for the Program.
583 |
584 | Later license versions may give you additional or different
585 | permissions. However, no additional obligations are imposed on any
586 | author or copyright holder as a result of your choosing to follow a
587 | later version.
588 |
589 | 15. Disclaimer of Warranty.
590 |
591 | THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY
592 | APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT
593 | HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY
594 | OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO,
595 | THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
596 | PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM
597 | IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF
598 | ALL NECESSARY SERVICING, REPAIR OR CORRECTION.
599 |
600 | 16. Limitation of Liability.
601 |
602 | IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
603 | WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MODIFIES AND/OR CONVEYS
604 | THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY
605 | GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE
606 | USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF
607 | DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD
608 | PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS),
609 | EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF
610 | SUCH DAMAGES.
611 |
612 | 17. Interpretation of Sections 15 and 16.
613 |
614 | If the disclaimer of warranty and limitation of liability provided
615 | above cannot be given local legal effect according to their terms,
616 | reviewing courts shall apply local law that most closely approximates
617 | an absolute waiver of all civil liability in connection with the
618 | Program, unless a warranty or assumption of liability accompanies a
619 | copy of the Program in return for a fee.
620 |
621 | END OF TERMS AND CONDITIONS
622 |
623 | How to Apply These Terms to Your New Programs
624 |
625 | If you develop a new program, and you want it to be of the greatest
626 | possible use to the public, the best way to achieve this is to make it
627 | free software which everyone can redistribute and change under these terms.
628 |
629 | To do so, attach the following notices to the program. It is safest
630 | to attach them to the start of each source file to most effectively
631 | state the exclusion of warranty; and each file should have at least
632 | the "copyright" line and a pointer to where the full notice is found.
633 |
634 | {one line to give the program's name and a brief idea of what it does.}
635 | Copyright (C) {year} {name of author}
636 |
637 | This program is free software: you can redistribute it and/or modify
638 | it under the terms of the GNU General Public License as published by
639 | the Free Software Foundation, either version 3 of the License, or
640 | (at your option) any later version.
641 |
642 | This program is distributed in the hope that it will be useful,
643 | but WITHOUT ANY WARRANTY; without even the implied warranty of
644 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
645 | GNU General Public License for more details.
646 |
647 | You should have received a copy of the GNU General Public License
648 | along with this program. If not, see .
649 |
650 | Also add information on how to contact you by electronic and paper mail.
651 |
652 | If the program does terminal interaction, make it output a short
653 | notice like this when it starts in an interactive mode:
654 |
655 | {project} Copyright (C) {year} {fullname}
656 | This program comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
657 | This is free software, and you are welcome to redistribute it
658 | under certain conditions; type `show c' for details.
659 |
660 | The hypothetical commands `show w' and `show c' should show the appropriate
661 | parts of the General Public License. Of course, your program's commands
662 | might be different; for a GUI interface, you would use an "about box".
663 |
664 | You should also get your employer (if you work as a programmer) or school,
665 | if any, to sign a "copyright disclaimer" for the program, if necessary.
666 | For more information on this, and how to apply and follow the GNU GPL, see
667 | .
668 |
669 | The GNU General Public License does not permit incorporating your program
670 | into proprietary programs. If your program is a subroutine library, you
671 | may consider it more useful to permit linking proprietary applications with
672 | the library. If this is what you want to do, use the GNU Lesser General
673 | Public License instead of this License. But first, please read
674 | .
675 |
--------------------------------------------------------------------------------