mirror of
https://github.com/eried/portapack-mayhem.git
synced 2024-10-01 01:26:06 -04:00
Wardrive map improv (#2251)
Added paginator. Added Flipper SUB file support.
This commit is contained in:
parent
21facd9cc3
commit
bfa7f3ca6b
@ -206,6 +206,7 @@ set(CPPSRC
|
|||||||
irq_rtc.cpp
|
irq_rtc.cpp
|
||||||
log_file.cpp
|
log_file.cpp
|
||||||
metadata_file.cpp
|
metadata_file.cpp
|
||||||
|
flipper_subfile.cpp
|
||||||
portapack.cpp
|
portapack.cpp
|
||||||
usb_serial_shell.cpp
|
usb_serial_shell.cpp
|
||||||
usb_serial_shell_filesystem.cpp
|
usb_serial_shell_filesystem.cpp
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
#include "string_format.hpp"
|
#include "string_format.hpp"
|
||||||
#include "file_path.hpp"
|
#include "file_path.hpp"
|
||||||
#include "metadata_file.hpp"
|
#include "metadata_file.hpp"
|
||||||
|
#include "flipper_subfile.hpp"
|
||||||
#include "portapack_persistent_memory.hpp"
|
#include "portapack_persistent_memory.hpp"
|
||||||
|
|
||||||
using namespace portapack;
|
using namespace portapack;
|
||||||
@ -36,14 +37,14 @@ void WardriveMapView::focus() {
|
|||||||
|
|
||||||
// needs to load on every map change, because won't store or draw all while marker is not in the current view.
|
// needs to load on every map change, because won't store or draw all while marker is not in the current view.
|
||||||
// todo optimize this somehow
|
// todo optimize this somehow
|
||||||
bool WardriveMapView::load_markers() {
|
void WardriveMapView::load_markers() {
|
||||||
uint16_t cnt = 0;
|
|
||||||
uint16_t displayed_cnt = 0;
|
uint16_t displayed_cnt = 0;
|
||||||
|
uint16_t cnt = 0;
|
||||||
geomap.clear_markers();
|
geomap.clear_markers();
|
||||||
// serach for files with geotag, and add it to geomap as marker with tag. limit to N bc of mem limit.
|
// serach for files with geotag, and add it to geomap as marker with tag. limit to N bc of mem limit.
|
||||||
for (const auto& entry : std::filesystem::directory_iterator(captures_dir, u"*.txt")) {
|
for (const auto& entry : std::filesystem::directory_iterator(captures_dir, u"*.txt")) {
|
||||||
if (std::filesystem::is_regular_file(entry.status())) {
|
if (std::filesystem::is_regular_file(entry.status())) {
|
||||||
if (displayed_cnt > 30) break;
|
if (markers_counted && displayed_cnt > ui::GeoMap::NumMarkerListElements) return; // only if not fist iteration, since then not counted all elements
|
||||||
std::filesystem::path pth = captures_dir;
|
std::filesystem::path pth = captures_dir;
|
||||||
pth += u"/" + entry.path();
|
pth += u"/" + entry.path();
|
||||||
auto metadata_path = get_metadata_path(pth);
|
auto metadata_path = get_metadata_path(pth);
|
||||||
@ -51,23 +52,65 @@ bool WardriveMapView::load_markers() {
|
|||||||
|
|
||||||
if (metadata) {
|
if (metadata) {
|
||||||
if (metadata.value().latitude != 0 && metadata.value().longitude != 0 && metadata.value().latitude < 400 && metadata.value().longitude < 400) {
|
if (metadata.value().latitude != 0 && metadata.value().longitude != 0 && metadata.value().latitude < 400 && metadata.value().longitude < 400) {
|
||||||
if (first_init == false) {
|
// skip nth
|
||||||
// move map there before add, so will display this.
|
if (marker_start <= cnt) {
|
||||||
geopos.set_report_change(false);
|
if (first_init == false) {
|
||||||
geopos.set_lat(metadata.value().latitude);
|
// move map there before add, so will display this.
|
||||||
geopos.set_lon(metadata.value().longitude);
|
geopos.set_report_change(false);
|
||||||
geopos.set_report_change(true);
|
geopos.set_lat(metadata.value().latitude);
|
||||||
geomap.move(metadata.value().longitude, metadata.value().latitude);
|
geopos.set_lon(metadata.value().longitude);
|
||||||
first_init = true;
|
geopos.set_report_change(true);
|
||||||
|
geomap.move(metadata.value().longitude, metadata.value().latitude);
|
||||||
|
first_init = true;
|
||||||
|
}
|
||||||
|
GeoMarker tmp{metadata.value().latitude, metadata.value().longitude, 400, entry.path().filename().string()};
|
||||||
|
if (geomap.store_marker(tmp) == MapMarkerStored::MARKER_STORED) displayed_cnt++;
|
||||||
|
if (!markers_counted) marker_cntall++;
|
||||||
}
|
}
|
||||||
GeoMarker tmp{metadata.value().latitude, metadata.value().longitude, 400, entry.path().filename().string()};
|
|
||||||
if (geomap.store_marker(tmp) == MapMarkerStored::MARKER_STORED) displayed_cnt++;
|
|
||||||
cnt++;
|
cnt++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return (cnt > 0);
|
// load flipper files too
|
||||||
|
for (const auto& entry : std::filesystem::directory_iterator(flippersub_dir, u"*.sub")) {
|
||||||
|
if (std::filesystem::is_regular_file(entry.status())) {
|
||||||
|
if (markers_counted && displayed_cnt > ui::GeoMap::NumMarkerListElements) return; // only if not fist iteration, since then not counted all elements
|
||||||
|
std::filesystem::path pth = flippersub_dir;
|
||||||
|
pth += u"/" + entry.path();
|
||||||
|
auto metadata = read_flippersub_file(pth);
|
||||||
|
|
||||||
|
if (metadata) {
|
||||||
|
if (metadata.value().latitude != 0 && metadata.value().longitude != 0 && metadata.value().latitude < 400 && metadata.value().longitude < 400) {
|
||||||
|
// skip nth
|
||||||
|
if (marker_start <= cnt) {
|
||||||
|
if (first_init == false) {
|
||||||
|
// move map there before add, so will display this.
|
||||||
|
geopos.set_report_change(false);
|
||||||
|
geopos.set_lat(metadata.value().latitude);
|
||||||
|
geopos.set_lon(metadata.value().longitude);
|
||||||
|
geopos.set_report_change(true);
|
||||||
|
geomap.move(metadata.value().longitude, metadata.value().latitude);
|
||||||
|
first_init = true;
|
||||||
|
}
|
||||||
|
GeoMarker tmp{metadata.value().latitude, metadata.value().longitude, 400, entry.path().filename().string()};
|
||||||
|
if (geomap.store_marker(tmp) == MapMarkerStored::MARKER_STORED) displayed_cnt++;
|
||||||
|
if (!markers_counted) marker_cntall++;
|
||||||
|
}
|
||||||
|
cnt++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
markers_counted = true;
|
||||||
|
// show / hide paginator buttons
|
||||||
|
btn_back.hidden((marker_start == 0) || (marker_cntall == 0));
|
||||||
|
btn_next.hidden(((marker_start + ui::GeoMap::NumMarkerListElements) >= marker_cntall) || (marker_cntall == 0));
|
||||||
|
// update text
|
||||||
|
text_info.set(to_string_dec_uint(marker_start + 1) + " - " + to_string_dec_uint(displayed_cnt + marker_start) + " / " + to_string_dec_uint(marker_cntall));
|
||||||
|
set_dirty();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
WardriveMapView::WardriveMapView(NavigationView& nav)
|
WardriveMapView::WardriveMapView(NavigationView& nav)
|
||||||
@ -75,7 +118,9 @@ WardriveMapView::WardriveMapView(NavigationView& nav)
|
|||||||
add_children({&text_info,
|
add_children({&text_info,
|
||||||
&geomap,
|
&geomap,
|
||||||
&geopos,
|
&geopos,
|
||||||
&text_notfound});
|
&text_notfound,
|
||||||
|
&btn_back,
|
||||||
|
&btn_next});
|
||||||
|
|
||||||
geomap.set_mode(DISPLAY);
|
geomap.set_mode(DISPLAY);
|
||||||
geomap.set_manual_panning(false);
|
geomap.set_manual_panning(false);
|
||||||
@ -100,21 +145,38 @@ WardriveMapView::WardriveMapView(NavigationView& nav)
|
|||||||
load_markers();
|
load_markers();
|
||||||
geomap.set_dirty();
|
geomap.set_dirty();
|
||||||
};
|
};
|
||||||
text_notfound.hidden(true);
|
|
||||||
geomap.init();
|
geomap.init();
|
||||||
// load markers
|
load_markers();
|
||||||
if (load_markers()) {
|
if (marker_cntall > 0) {
|
||||||
text_notfound.hidden(true);
|
text_notfound.hidden(true);
|
||||||
geomap.set_dirty();
|
geomap.set_dirty();
|
||||||
} else {
|
} else {
|
||||||
geomap.hidden(true);
|
geomap.hidden(true);
|
||||||
geopos.hidden(true);
|
geopos.hidden(true);
|
||||||
|
text_notfound.hidden(false);
|
||||||
|
text_info.hidden(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// never move this before the first load() bc that will mess load up
|
||||||
geomap.on_move = [this](float lon, float lat) {
|
geomap.on_move = [this](float lon, float lat) {
|
||||||
(void)lon;
|
(void)lon;
|
||||||
(void)lat;
|
(void)lat;
|
||||||
load_markers();
|
load_markers();
|
||||||
};
|
};
|
||||||
|
btn_back.on_select = [this, &nav](Button&) {
|
||||||
|
if (marker_start - ui::GeoMap::NumMarkerListElements >= 0)
|
||||||
|
marker_start = marker_start - ui::GeoMap::NumMarkerListElements;
|
||||||
|
else
|
||||||
|
marker_start = 0;
|
||||||
|
load_markers();
|
||||||
|
};
|
||||||
|
btn_next.on_select = [this, &nav](Button&) {
|
||||||
|
if (marker_start + ui::GeoMap::NumMarkerListElements <= marker_cntall)
|
||||||
|
marker_start = marker_start + ui::GeoMap::NumMarkerListElements;
|
||||||
|
else
|
||||||
|
marker_start = marker_cntall - ui::GeoMap::NumMarkerListElements;
|
||||||
|
load_markers();
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
WardriveMapView::~WardriveMapView() {
|
WardriveMapView::~WardriveMapView() {
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2014 Jared Boone, ShareBrained Technology, Inc.
|
* Copyright (C) 2024 HTotoo
|
||||||
* Copyright (C) 2017 Furrtek
|
|
||||||
*
|
*
|
||||||
* This file is part of PortaPack.
|
* This file is part of PortaPack.
|
||||||
*
|
*
|
||||||
@ -20,10 +19,6 @@
|
|||||||
* Boston, MA 02110-1301, USA.
|
* Boston, MA 02110-1301, USA.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// Code from https://github.com/Flipper-XFW/Xtreme-Apps/tree/04c3a60093e2c2378e79498b4505aa8072980a42/ble_spam/protocols
|
|
||||||
// Thanks for the work of the original creators!
|
|
||||||
// Saying thanks in the main view!
|
|
||||||
|
|
||||||
#ifndef __UI_WARDRIVEMAP_H__
|
#ifndef __UI_WARDRIVEMAP_H__
|
||||||
#define __UI_WARDRIVEMAP_H__
|
#define __UI_WARDRIVEMAP_H__
|
||||||
|
|
||||||
@ -50,22 +45,29 @@ class WardriveMapView : public View {
|
|||||||
};
|
};
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
const std::filesystem::path flippersub_dir = u"subghz";
|
||||||
NavigationView& nav_;
|
NavigationView& nav_;
|
||||||
|
|
||||||
Text text_info{{0 * 8, 0 * 8, 30 * 8, 16 * 1}, "All GEOTAG from CAPTURES"};
|
Text text_info{{0 * 8, 0 * 8, 20 * 8, 16 * 1}, "0 / 30"};
|
||||||
Text text_notfound{{0 * 8, 3 * 8, 30 * 8, 16 * 1}, "No GeoTagged captures found"};
|
Text text_notfound{{0 * 8, 0 * 8, 30 * 8, 16 * 1}, "No GeoTagged captures found"};
|
||||||
GeoPos geopos{
|
GeoPos geopos{
|
||||||
{0, 20},
|
{0, 20},
|
||||||
GeoPos::alt_unit::METERS,
|
GeoPos::alt_unit::METERS,
|
||||||
GeoPos::spd_unit::HIDDEN};
|
GeoPos::spd_unit::HIDDEN};
|
||||||
GeoMap geomap{{0, 75, 240, 320 - 75}};
|
GeoMap geomap{{0, 75, 240, 320 - 75}};
|
||||||
|
|
||||||
|
Button btn_back{{22 * 8, 0 * 8, 3 * 8, 16}, "<-"};
|
||||||
|
Button btn_next{{26 * 8, 0 * 8, 3 * 8, 16}, "->"};
|
||||||
|
|
||||||
void on_gps(const GPSPosDataMessage* msg);
|
void on_gps(const GPSPosDataMessage* msg);
|
||||||
void on_orientation(const OrientationDataMessage* msg);
|
void on_orientation(const OrientationDataMessage* msg);
|
||||||
|
|
||||||
bool load_markers(); // returns true if any exists, false if none.
|
void load_markers(); // returns true if any exists, false if none.
|
||||||
|
|
||||||
bool first_init = false;
|
bool first_init = false; // to center map to first marker, before callback is set
|
||||||
|
bool markers_counted = false; // to iterate all files on first, but only on first.
|
||||||
|
uint16_t marker_start = 0; // for paginator, this will be the first displayed
|
||||||
|
uint16_t marker_cntall = 0; // all geotagged marker count
|
||||||
|
|
||||||
MessageHandlerRegistration message_handler_gps{
|
MessageHandlerRegistration message_handler_gps{
|
||||||
Message::ID::GPSPosData,
|
Message::ID::GPSPosData,
|
||||||
|
117
firmware/application/flipper_subfile.cpp
Normal file
117
firmware/application/flipper_subfile.cpp
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2024 HTotoo
|
||||||
|
*
|
||||||
|
* This file is part of PortaPack.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2, or (at your option)
|
||||||
|
* any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; see the file COPYING. If not, write to
|
||||||
|
* the Free Software Foundation, Inc., 51 Franklin Street,
|
||||||
|
* Boston, MA 02110-1301, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "flipper_subfile.hpp"
|
||||||
|
|
||||||
|
#include "convert.hpp"
|
||||||
|
#include "file_reader.hpp"
|
||||||
|
#include "string_format.hpp"
|
||||||
|
#include <string_view>
|
||||||
|
|
||||||
|
namespace fs = std::filesystem;
|
||||||
|
using namespace std::literals;
|
||||||
|
|
||||||
|
const std::string_view filetype_name = "Filetype"sv;
|
||||||
|
const std::string_view frequency_name = "Frequency"sv;
|
||||||
|
const std::string_view latitude_name = "Latitute"sv;
|
||||||
|
const std::string_view longitude_name = "Longitude"sv;
|
||||||
|
const std::string_view protocol_name = "Protocol"sv;
|
||||||
|
const std::string_view preset_name = "Preset"sv;
|
||||||
|
const std::string_view te_name = "TE"sv; // only in BinRAW
|
||||||
|
|
||||||
|
/*
|
||||||
|
Filetype: Flipper SubGhz Key File
|
||||||
|
Version: 1
|
||||||
|
Frequency: 433920000
|
||||||
|
Preset: FuriHalSubGhzPresetOok650Async
|
||||||
|
Latitute: nan
|
||||||
|
Longitude: nan
|
||||||
|
Protocol: BinRAW
|
||||||
|
Bit: 1730
|
||||||
|
TE: 495
|
||||||
|
Bit_RAW: 1730
|
||||||
|
Data_RAW: 02 10 84
|
||||||
|
|
||||||
|
te: is the quantization interval, in us.
|
||||||
|
bit: all bit counts in file.
|
||||||
|
bit_raw: the bits stored in the next data_raw. this 2 can repeat
|
||||||
|
data_raw: is an encoded sequence of durations, where each bit in the sequence encodes one TE interval: 1 - high level (there is a carrier), 0 - low (no carrier). For example, TE=100, Bit_RAW=8, Data_RAW=0x37 => 0b00110111, that is, -200 200 -100 300 will be transmitted
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Filetype: Flipper SubGhz RAW File
|
||||||
|
Version: 1
|
||||||
|
Frequency: 433920000
|
||||||
|
Preset: FuriHalSubGhzPresetOok650Async
|
||||||
|
Protocol: RAW
|
||||||
|
RAW_Data: 5832 -12188 130 -162
|
||||||
|
|
||||||
|
raw_data- positive: carrier for n time, negative: no carrier for n time. (us)
|
||||||
|
*/
|
||||||
|
|
||||||
|
Optional<flippersub_metadata> read_flippersub_file(const fs::path& path) {
|
||||||
|
File f;
|
||||||
|
auto error = f.open(path);
|
||||||
|
|
||||||
|
if (error)
|
||||||
|
return {};
|
||||||
|
|
||||||
|
flippersub_metadata metadata{};
|
||||||
|
|
||||||
|
auto reader = FileLineReader(f);
|
||||||
|
for (const auto& line : reader) {
|
||||||
|
auto cols = split_string(line, ':');
|
||||||
|
|
||||||
|
if (cols.size() != 2)
|
||||||
|
continue; // Bad line.
|
||||||
|
if (cols[1].length() <= 1) continue;
|
||||||
|
std::string fixed = cols[1].data() + 1;
|
||||||
|
fixed = trim(fixed);
|
||||||
|
if (cols[0] == filetype_name) {
|
||||||
|
if (fixed != "Flipper SubGhz Key File" && fixed != "Flipper SubGhz RAW File") return {}; // not supported
|
||||||
|
} else if (cols[0] == frequency_name)
|
||||||
|
parse_int(fixed, metadata.center_frequency);
|
||||||
|
else if (cols[0] == latitude_name)
|
||||||
|
parse_float_meta(fixed, metadata.latitude);
|
||||||
|
else if (cols[0] == longitude_name)
|
||||||
|
parse_float_meta(fixed, metadata.longitude);
|
||||||
|
else if (cols[0] == protocol_name) {
|
||||||
|
if (fixed == "RAW") metadata.protocol = FLIPPER_PROTO_RAW;
|
||||||
|
if (fixed == "BinRAW") metadata.protocol = FLIPPER_PROTO_BINRAW;
|
||||||
|
} else if (cols[0] == te_name) {
|
||||||
|
metadata.te = atoi(fixed.c_str());
|
||||||
|
} else if (cols[0] == preset_name) {
|
||||||
|
if (fixed.find("FSK") != std::string::npos) {
|
||||||
|
metadata.preset = FLIPPER_PRESET_2FSK;
|
||||||
|
} else if (fixed.find("Ook") != std::string::npos) {
|
||||||
|
metadata.preset = FLIPPER_PRESET_OOK;
|
||||||
|
} else if (fixed.find("Custom") != std::string::npos) {
|
||||||
|
metadata.preset = FLIPPER_PRESET_CUSTOM;
|
||||||
|
}
|
||||||
|
|
||||||
|
} else
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (metadata.center_frequency == 0) return {}; // Parse failed.
|
||||||
|
|
||||||
|
return metadata;
|
||||||
|
}
|
53
firmware/application/flipper_subfile.hpp
Normal file
53
firmware/application/flipper_subfile.hpp
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2024 HTotoo
|
||||||
|
*
|
||||||
|
* This file is part of PortaPack.
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation; either version 2, or (at your option)
|
||||||
|
* any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; see the file COPYING. If not, write to
|
||||||
|
* the Free Software Foundation, Inc., 51 Franklin Street,
|
||||||
|
* Boston, MA 02110-1301, USA.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __FLIPPER_SUBFILE_HPP__
|
||||||
|
#define __FLIPPER_SUBFILE_HPP__
|
||||||
|
|
||||||
|
#include "metadata_file.hpp"
|
||||||
|
|
||||||
|
typedef enum : uint8_t {
|
||||||
|
FLIPPER_PROTO_UNSUPPORTED = 0,
|
||||||
|
FLIPPER_PROTO_RAW = 1,
|
||||||
|
FLIPPER_PROTO_BINRAW = 2
|
||||||
|
} FlipperProto;
|
||||||
|
|
||||||
|
typedef enum : uint8_t {
|
||||||
|
FLIPPER_PRESET_UNK = 0,
|
||||||
|
FLIPPER_PRESET_CUSTOM = 1,
|
||||||
|
FLIPPER_PRESET_OOK = 2,
|
||||||
|
FLIPPER_PRESET_2FSK = 3,
|
||||||
|
} FlipperPreset;
|
||||||
|
|
||||||
|
struct flippersub_metadata {
|
||||||
|
rf::Frequency center_frequency;
|
||||||
|
float latitude = 0;
|
||||||
|
float longitude = 0;
|
||||||
|
FlipperProto protocol = FLIPPER_PROTO_UNSUPPORTED;
|
||||||
|
FlipperPreset preset = FLIPPER_PRESET_UNK;
|
||||||
|
uint16_t te = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
Optional<flippersub_metadata> read_flippersub_file(const std::filesystem::path& path);
|
||||||
|
|
||||||
|
// Maybe sometime there will be a data part reader / converter
|
||||||
|
|
||||||
|
#endif // __FLIPPER_SUBFILE_HPP__
|
Loading…
Reference in New Issue
Block a user