portapack-mayhem/firmware/application/tpms_app.hpp

236 lines
4.6 KiB
C++
Raw Normal View History

2015-12-02 00:30:52 -05:00
/*
* Copyright (C) 2015 Jared Boone, ShareBrained Technology, Inc.
*
* 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 __TPMS_APP_H__
#define __TPMS_APP_H__
2015-12-02 00:30:52 -05:00
2016-01-25 01:38:45 -05:00
#include "ui_widget.hpp"
#include "ui_navigation.hpp"
#include "field_reader.hpp"
#include "baseband_packet.hpp"
2015-12-02 00:30:52 -05:00
#include "manchester.hpp"
#include "log_file.hpp"
2015-12-02 00:30:52 -05:00
#include "recent_entries.hpp"
2016-01-18 16:54:07 -05:00
2016-01-24 00:26:06 -05:00
#include "optional.hpp"
2016-01-19 00:42:26 -05:00
#include "units.hpp"
using units::Temperature;
using units::Pressure;
2016-01-22 16:41:01 -05:00
namespace tpms {
class TransponderID {
public:
constexpr TransponderID(
) : id_ { 0 }
{
}
constexpr TransponderID(
const uint32_t id
) : id_ { id }
{
}
constexpr uint32_t value() const {
return id_;
}
private:
uint32_t id_;
};
2016-01-19 00:42:26 -05:00
class Reading {
public:
enum Type {
None = 0,
FLM_64 = 1,
FLM_72 = 2,
FLM_80 = 3,
};
2016-01-19 00:42:26 -05:00
constexpr Reading(
) : type_ { Type::None }
2016-01-19 00:42:26 -05:00
{
}
constexpr Reading(
Type type,
TransponderID id
) : type_ { type },
id_ { id }
2016-01-19 00:42:26 -05:00
{
}
2016-01-19 00:42:26 -05:00
constexpr Reading(
Type type,
TransponderID id,
Optional<Pressure> pressure = { },
Optional<Temperature> temperature = { }
) : type_ { type },
id_ { id },
pressure_ { pressure },
temperature_ { temperature }
2016-01-19 00:42:26 -05:00
{
}
Type type() const {
return type_;
}
TransponderID id() const {
2016-01-19 00:42:26 -05:00
return id_;
}
Optional<Pressure> pressure() const {
return pressure_;
2016-01-19 00:42:26 -05:00
}
Optional<Temperature> temperature() const {
return temperature_;
2016-01-19 00:42:26 -05:00
}
private:
Type type_ { Type::None };
TransponderID id_ { 0 };
Optional<Pressure> pressure_ { };
Optional<Temperature> temperature_ { };
2016-01-19 00:42:26 -05:00
};
2016-01-18 17:34:30 -05:00
class Packet {
public:
constexpr Packet(
const baseband::Packet& packet
) : packet_ { packet },
decoder_ { packet_, 0 },
reader_ { decoder_ }
2016-01-18 17:34:30 -05:00
{
}
Timestamp received_at() const;
ManchesterFormatted symbols_formatted() const;
2016-01-19 00:42:26 -05:00
Optional<Reading> reading() const;
2016-01-18 17:34:30 -05:00
private:
using Reader = FieldReader<ManchesterDecoder, BitRemapNone>;
2016-01-18 17:34:30 -05:00
const baseband::Packet packet_;
const ManchesterDecoder decoder_;
const Reader reader_;
2016-01-19 00:42:26 -05:00
size_t crc_valid_length() const;
2016-01-18 17:34:30 -05:00
};
2016-01-18 16:54:07 -05:00
} /* namespace tpms */
namespace std {
constexpr bool operator==(const tpms::TransponderID& lhs, const tpms::TransponderID& rhs) {
return (lhs.value() == rhs.value());
}
} /* namespace std */
struct TPMSRecentEntry {
using Key = std::pair<tpms::Reading::Type, tpms::TransponderID>;
static const Key invalid_key;
tpms::Reading::Type type { invalid_key.first };
tpms::TransponderID id { invalid_key.second };
size_t received_count { 0 };
Optional<Pressure> last_pressure;
Optional<Temperature> last_temperature;
TPMSRecentEntry(
const Key& key
) : type { key.first },
id { key.second }
{
}
Key key() const {
return { type, id };
}
void update(const tpms::Reading& reading);
};
using TPMSRecentEntries = RecentEntries<tpms::Reading, TPMSRecentEntry>;
class TPMSLogger {
2015-12-02 00:30:52 -05:00
public:
TPMSLogger(const std::string& file_path);
void on_packet(const tpms::Packet& packet, const uint32_t target_frequency);
2015-12-02 00:30:52 -05:00
private:
LogFile log_file;
2015-12-02 00:30:52 -05:00
};
namespace ui {
using TPMSRecentEntriesView = RecentEntriesView<TPMSRecentEntries>;
2016-01-18 16:41:19 -05:00
class TPMSAppView : public View {
2015-12-02 00:30:52 -05:00
public:
TPMSAppView(NavigationView& nav);
2016-01-18 16:41:19 -05:00
~TPMSAppView();
void set_parent_rect(const Rect new_parent_rect) override;
2015-12-02 00:30:52 -05:00
// Prevent painting of region covered entirely by a child.
// TODO: Add flag to View that specifies view does not need to be cleared before painting.
void paint(Painter&) override { };
2016-01-27 13:18:44 -05:00
void focus() override;
2016-01-26 16:08:46 -05:00
std::string title() const override { return "TPMS"; };
2015-12-02 00:30:52 -05:00
private:
static constexpr uint32_t initial_target_frequency = 315000000;
static constexpr uint32_t sampling_rate = 2457600;
static constexpr uint32_t baseband_bandwidth = 1750000;
TPMSRecentEntries recent;
std::unique_ptr<TPMSLogger> logger;
2015-12-02 00:30:52 -05:00
TPMSRecentEntriesView recent_entries_view { recent };
2016-01-18 16:41:19 -05:00
2016-01-18 17:34:30 -05:00
void on_packet(const tpms::Packet& packet);
void on_show_list();
uint32_t target_frequency() const;
uint32_t tuning_frequency() const;
2015-12-02 00:30:52 -05:00
};
} /* namespace ui */
#endif/*__TPMS_APP_H__*/