/* ; Project: Open Vehicle Monitor System ; Module: CAN logging framework ; Date: 18th January 2018 ; ; (C) 2018 Michael Balzer ; ; Permission is hereby granted, free of charge, to any person obtaining a copy ; of this software and associated documentation files (the "Software"), to deal ; in the Software without restriction, including without limitation the rights ; to use, copy, modify, merge, publish, distribute, sublicense, and/or sell ; copies of the Software, and to permit persons to whom the Software is ; furnished to do so, subject to the following conditions: ; ; The above copyright notice and this permission notice shall be included in ; all copies or substantial portions of the Software. ; ; THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR ; IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, ; FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE ; AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER ; LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, ; OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN ; THE SOFTWARE. */ #include "ovms_log.h" static const char *TAG = "canlog-monitor"; #include "can.h" #include "canlog_monitor.h" void can_log_monitor_start(int verbosity, OvmsWriter* writer, OvmsCommand* cmd, int argc, const char* const* argv) { std::string format(cmd->GetName()); canlog_monitor* logger = new canlog_monitor(format); logger->Open(); MyCan.AddLogger(logger, argc, argv); writer->printf("CAN logging to MONITOR active: %s\n", logger->GetInfo().c_str()); writer->puts("Note: info logging is at debug log level, frame logging is at verbose, and errors as usual"); } class OvmsCanLogMonitorInit { public: OvmsCanLogMonitorInit(); } MyOvmsCanLogMonitorInit __attribute__ ((init_priority (4560))); OvmsCanLogMonitorInit::OvmsCanLogMonitorInit() { ESP_LOGI(TAG, "Initialising CAN logging to MONITOR (4560)"); OvmsCommand* cmd_can = MyCommandApp.FindCommand("can"); if (cmd_can) { OvmsCommand* cmd_can_log = cmd_can->FindCommand("log"); if (cmd_can_log) { OvmsCommand* cmd_can_log_start = cmd_can_log->FindCommand("start"); if (cmd_can_log_start) { // We have a place to put our command tree.. OvmsCommand* start = cmd_can_log_start->RegisterCommand("monitor", "CAN logging to MONITOR"); MyCanFormatFactory.RegisterCommandSet(start, "Start CAN logging to MONITOR", can_log_monitor_start, "[filter1] ... [filterN]\n" "Filter: | [-] | :[-]\n" "Example: 2:2a0-37f", 0, 9); } } } } canlog_monitor_conn::canlog_monitor_conn(canlog* logger, std::string format, canformat::canformat_serve_mode_t mode) : canlogconnection(logger, format, mode) { } canlog_monitor_conn::~canlog_monitor_conn() { } void canlog_monitor_conn::OutputMsg(CAN_log_message_t& msg, std::string &result) { m_msgcount++; if ((m_filters != NULL) && (! m_filters->IsFiltered(&msg.frame))) { m_filtercount++; return; } if (result.length()>0) { switch (msg.type) { case CAN_LogFrame_RX: case CAN_LogFrame_TX: case CAN_LogFrame_TX_Queue: case CAN_LogFrame_TX_Fail: ESP_LOGV(TAG,"%s",result.c_str()); break; case CAN_LogStatus_Error: ESP_LOGE(TAG,"%s",result.c_str()); break; case CAN_LogStatus_Statistics: case CAN_LogInfo_Comment: case CAN_LogInfo_Config: case CAN_LogInfo_Event: ESP_LOGD(TAG,"%s",result.c_str()); break; default: break; } } } canlog_monitor::canlog_monitor(std::string format) : canlog("monitor", format) { } canlog_monitor::~canlog_monitor() { } bool canlog_monitor::Open() { ESP_LOGI(TAG, "Now logging CAN messages to monitor"); OvmsRecMutexLock lock(&m_cmmutex); canlog_monitor_conn* clc = new canlog_monitor_conn(this, m_format, m_mode); clc->m_peer = std::string("MONITOR"); m_connmap[NULL] = clc; m_isopen = true; std::string header = m_formatter->getheader(); if (header.length()>0) { ESP_LOGD(TAG,"%s",header.c_str()); } return true; } void canlog_monitor::Close() { ESP_LOGI(TAG, "Closed monitor log: %s", GetStats().c_str()); OvmsRecMutexLock lock(&m_cmmutex); for (conn_map_t::iterator it=m_connmap.begin(); it!=m_connmap.end(); ++it) { delete it->second; } m_connmap.clear(); m_isopen = false; }