/* * 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. */ #include "tpms_app.hpp" #include "event_m0.hpp" #include "baseband_api.hpp" #include "string_format.hpp" #include "crc.hpp" #include "utility.hpp" namespace tpms { namespace format { std::string type(Reading::Type type) { return to_string_dec_uint(toUType(type), 2); } std::string id(TransponderID id) { return to_string_hex(id.value(), 8); } std::string pressure(Pressure pressure) { return to_string_dec_int(pressure.kilopascal(), 3); } std::string temperature(Temperature temperature) { return to_string_dec_int(temperature.celsius(), 3); } } /* namespace format */ Timestamp Packet::received_at() const { return packet_.timestamp(); } ManchesterFormatted Packet::symbols_formatted() const { return format_manchester(decoder_); } Optional Packet::reading() const { const auto length = crc_valid_length(); switch(length) { case 64: return Reading { Reading::Type::FLM_64, reader_.read(0, 32), Pressure { static_cast(reader_.read(32, 8)) * 4 / 3 }, Temperature { static_cast(reader_.read(40, 8) & 0x7f) - 50 } }; case 72: return Reading { Reading::Type::FLM_72, reader_.read(0, 32), Pressure { static_cast(reader_.read(40, 8)) * 4 / 3 }, Temperature { static_cast(reader_.read(48, 8)) - 50 } }; case 80: return Reading { Reading::Type::FLM_80, reader_.read(8, 32), Pressure { static_cast(reader_.read(48, 8)) * 4 / 3 }, Temperature { static_cast(reader_.read(56, 8)) - 50 } }; default: return { }; } } size_t Packet::crc_valid_length() const { constexpr uint32_t checksum_bytes = 0b1111111; constexpr uint32_t crc_72_bytes = 0b111111111; constexpr uint32_t crc_80_bytes = 0b1111111110; std::array bytes; for(size_t i=0; i crc_72 { 0x01, 0x00 }; CRC crc_80 { 0x01, 0x00 }; for(size_t i=0; i= 1<<32, weirdness will ensue! const auto tuning_frequency_str = to_string_dec_uint(target_frequency, 10); std::string entry = tuning_frequency_str + " FSK 38.4 19.2 " + hex_formatted.data + "/" + hex_formatted.errors; log_file.write_entry(packet.received_at(), entry); } } const TPMSRecentEntry::Key TPMSRecentEntry::invalid_key = { tpms::Reading::Type::None, 0 }; void TPMSRecentEntry::update(const tpms::Reading& reading) { received_count++; if( reading.pressure().is_valid() ) { last_pressure = reading.pressure(); } if( reading.temperature().is_valid() ) { last_temperature = reading.temperature(); } } namespace ui { static const std::array, 5> tpms_columns { { { "Tp", 2 }, { "ID", 8 }, { "kPa", 3 }, { "C", 3 }, { "Cnt", 3 }, } }; template<> void RecentEntriesView::draw_header( const Rect& target_rect, Painter& painter, const Style& style ) { auto x = 0; for(const auto& column : tpms_columns) { const auto width = column.second; auto text = column.first; if( width > text.length() ) { text.append(width - text.length(), ' '); } painter.draw_string({ x, target_rect.pos.y }, style, text); x += (width * 8) + 8; } } template<> void RecentEntriesView::draw( const Entry& entry, const Rect& target_rect, Painter& painter, const Style& style, const bool is_selected ) { const auto& draw_style = is_selected ? style.invert() : style; std::string line = tpms::format::type(entry.type) + " " + tpms::format::id(entry.id); if( entry.last_pressure.is_valid() ) { line += " " + tpms::format::pressure(entry.last_pressure.value()); } else { line += " " " "; } if( entry.last_temperature.is_valid() ) { line += " " + tpms::format::temperature(entry.last_temperature.value()); } else { line += " " " "; } if( entry.received_count > 999 ) { line += " +++"; } else { line += " " + to_string_dec_uint(entry.received_count, 3); } line.resize(target_rect.width() / 8, ' '); painter.draw_string(target_rect.pos, draw_style, line); } TPMSAppView::TPMSAppView(NavigationView&) { add_children({ { &recent_entries_view, } }); EventDispatcher::message_map().register_handler(Message::ID::TPMSPacket, [this](Message* const p) { const auto message = static_cast(p); const tpms::Packet packet { message->packet }; this->on_packet(packet); } ); radio::enable({ tuning_frequency(), sampling_rate, baseband_bandwidth, rf::Direction::Receive, false, 32, 32, 1, }); baseband::start({ .mode = 5, .sampling_rate = sampling_rate, .decimation_factor = 1, }); logger = std::make_unique("tpms.txt"); } TPMSAppView::~TPMSAppView() { baseband::stop(); radio::disable(); EventDispatcher::message_map().unregister_handler(Message::ID::TPMSPacket); } void TPMSAppView::focus() { recent_entries_view.focus(); } void TPMSAppView::set_parent_rect(const Rect new_parent_rect) { View::set_parent_rect(new_parent_rect); recent_entries_view.set_parent_rect({ 0, 0, new_parent_rect.width(), new_parent_rect.height() }); } void TPMSAppView::on_packet(const tpms::Packet& packet) { if( logger ) { logger->on_packet(packet, target_frequency()); } const auto reading_opt = packet.reading(); if( reading_opt.is_valid() ) { const auto reading = reading_opt.value(); recent.on_packet({ reading.type(), reading.id() }, reading); recent_entries_view.set_dirty(); } } void TPMSAppView::on_show_list() { recent_entries_view.hidden(false); recent_entries_view.focus(); } uint32_t TPMSAppView::target_frequency() const { return initial_target_frequency; } uint32_t TPMSAppView::tuning_frequency() const { return target_frequency() - (sampling_rate / 4); } } /* namespace ui */