/* 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 3 of the License, 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. If not, see . */ #include #if HAL_WITH_UAVCAN #include "AP_Compass_UAVCAN.h" #if HAL_OS_POSIX_IO #include #include #include #include #endif #include #include extern const AP_HAL::HAL& hal; #define debug_mag_uavcan(level, fmt, args...) do { if ((level) <= AP_BoardConfig_CAN::get_can_debug()) { printf(fmt, ##args); }} while (0) /* constructor - registers instance at top Compass driver */ AP_Compass_UAVCAN::AP_Compass_UAVCAN(Compass &compass): AP_Compass_Backend(compass) { _sem_mag = hal.util->new_semaphore(); } AP_Compass_UAVCAN::~AP_Compass_UAVCAN() { if (!_initialized) { return; } AP_UAVCAN *ap_uavcan = AP_UAVCAN::get_uavcan(_manager); if (ap_uavcan == nullptr) { return; } ap_uavcan->remove_mag_listener(this); delete _sem_mag; debug_mag_uavcan(2, "AP_Compass_UAVCAN destructed\n\r"); } AP_Compass_Backend *AP_Compass_UAVCAN::probe(Compass &compass) { if (AP_BoardConfig_CAN::get_can_num_ifaces() == 0) { return nullptr; } AP_Compass_UAVCAN *sensor; for (uint8_t i = 0; i < MAX_NUMBER_OF_CAN_DRIVERS; i++) { AP_UAVCAN *ap_uavcan = AP_UAVCAN::get_uavcan(i); if (ap_uavcan == nullptr) { continue; } uint8_t freemag = ap_uavcan->find_smallest_free_mag_node(); if (freemag == UINT8_MAX) { continue; } sensor = new AP_Compass_UAVCAN(compass); if (sensor->register_uavcan_compass(i, freemag)) { debug_mag_uavcan(2, "AP_Compass_UAVCAN probed, drv: %d, node: %d\n\r", i, freemag); return sensor; } else { delete sensor; } } return nullptr; } bool AP_Compass_UAVCAN::register_uavcan_compass(uint8_t mgr, uint8_t node) { AP_UAVCAN *ap_uavcan = AP_UAVCAN::get_uavcan(mgr); if (ap_uavcan == nullptr) { return false; } _manager = mgr; if (ap_uavcan->register_mag_listener_to_node(this, node)) { _instance = register_compass(); struct DeviceStructure { uint8_t bus_type : 3; uint8_t bus: 5; uint8_t address; uint8_t devtype; }; union DeviceId { struct DeviceStructure devid_s; uint32_t devid; }; union DeviceId d; d.devid_s.bus_type = 3; d.devid_s.bus = mgr; d.devid_s.address = node; d.devid_s.devtype = 1; set_dev_id(_instance, d.devid); set_external(_instance, true); _sum.zero(); _count = 0; debug_mag_uavcan(2, "AP_Compass_UAVCAN loaded\n\r"); return true; } return false; } void AP_Compass_UAVCAN::read(void) { // avoid division by zero if we haven't received any mag reports if (_count == 0) { return; } if (_sem_mag->take(HAL_SEMAPHORE_BLOCK_FOREVER)) { _sum /= _count; publish_filtered_field(_sum, _instance); _sum.zero(); _count = 0; _sem_mag->give(); } } void AP_Compass_UAVCAN::handle_mag_msg(Vector3f &mag) { Vector3f raw_field = mag * 1000.0; // rotate raw_field from sensor frame to body frame rotate_field(raw_field, _instance); // publish raw_field (uncorrected point sample) for calibration use publish_raw_field(raw_field, _instance); // correct raw_field for known errors correct_field(raw_field, _instance); if (_sem_mag->take(HAL_SEMAPHORE_BLOCK_FOREVER)) { // accumulate into averaging filter _sum += raw_field; _count++; _sem_mag->give(); } } #endif