// // The MIT License (MIT) // // Copyright (c) 2019 Livox. All rights reserved. // // 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 "lds_lidar.h" #include #include #include #include namespace livox { /** Const varible ------------------------------------------------------------------------------- */ /** User add broadcast code here */ LidarDevice lidars_[kMaxLidarCount]; bool is_initialized_=false; bool is_created=false; thread_safe_map g_snPtr; static const char *local_broadcast_code_list[] = { "000000000000001", }; /** Lds lidar function ---------------------------------------------------------------------------*/ LdsLidar::LdsLidar() { if(is_created==false) { auto_connect_mode_ = true; whitelist_count_ = 0; is_initialized_ = false; lidar_count_ = 0; memset(broadcast_code_whitelist_, 0, sizeof(broadcast_code_whitelist_)); memset(lidars_, 0, sizeof(lidars_)); for (uint32_t i = 0; i < kMaxLidarCount; i++) { lidars_[i].handle = kMaxLidarCount; /** Unallocated state */ lidars_[i].connect_state = kConnectStateOff; } is_created=true; } } LdsLidar::~LdsLidar() { } int LdsLidar::AddBroadcastCode(std::string sn) { if(g_snPtr.find(sn)==true) { printf(" %s has added \n",sn.c_str()); return -1; } g_snPtr[sn]=this; return 0; } void LdsLidar::LidarDataCallback(uint8_t handle, LivoxEthPacket *data,uint32_t data_num) { printf("data recieved ------------------------------------ \n"); } int LdsLidar::InitLdsLidar() { if (is_initialized_==false) { if (!Init()) { Uninit(); printf("Livox-SDK init fail!\n"); return -1; } LivoxSdkVersion _sdkversion; GetLivoxSdkVersion(&_sdkversion); printf("Livox SDK version %d.%d.%d\n", _sdkversion.major, _sdkversion.minor, _sdkversion.patch); } SetBroadcastCallback(LdsLidar::OnDeviceBroadcast); SetDeviceStateUpdateCallback(LdsLidar::OnDeviceChange); /** Add commandline input broadcast code */ /*for (auto input_str : broadcast_code_strs) { LdsLidar::AddBroadcastCodeToWhitelist(input_str.c_str()); }*/ /** Add local broadcast code */ /*LdsLidar::AddLocalBroadcastCode(); if (whitelist_count_) { LdsLidar::DisableAutoConnectMode(); printf("Disable auto connect mode!\n"); printf("List all broadcast code in whiltelist:\n"); for (uint32_t i = 0; i < whitelist_count_; i++) { printf("%s\n", broadcast_code_whitelist_[i]); } } else { LdsLidar::EnableAutoConnectMode(); printf("No broadcast code was added to whitelist, swith to automatic connection mode!\n"); }*/ /** Start livox sdk to receive lidar data */ if (!Start()) { Uninit(); printf("Livox-SDK init fail!\n"); return -1; } printf(" start -----------\n"); /** Add here, only for callback use */ is_initialized_ = true; printf("Livox-SDK init success!\n"); return 0; } int LdsLidar::DeInitLdsLidar(void) { if (!is_initialized_) { printf("LiDAR data source is not exit"); return -1; } Uninit(); printf("Livox SDK Deinit completely!\n"); return 0; } /** Static function in LdsLidar for callback or event process ------------------------------------*/ /** Receiving point cloud data from Livox LiDAR. */ void LdsLidar::GetLidarDataCb(uint8_t handle, LivoxEthPacket *data, uint32_t data_num, void *client_data) { using namespace std; LdsLidar *lidar_this = static_cast(client_data); //printf("receive handle %d ptr %p\n", handle,client_data); if(lidar_this!= nullptr) lidar_this->LidarDataCallback(handle,data,data_num); /*LivoxEthPacket *eth_packet = data; if (!data || !data_num || (handle >= kMaxLidarCount)) { return; } if (eth_packet) { lidar_this->data_recveive_count_[handle]++; if (lidar_this->data_recveive_count_[handle] % 100 == 0) { printf("receive packet count %d %d\n", handle, lidar_this->data_recveive_count_[handle]); uint64_t cur_timestamp = *((uint64_t *) (data->timestamp)); if (data->data_type == kCartesian) { printf("1\n"); LivoxRawPoint *p_point_data = (LivoxRawPoint *) data->data; } else if (data->data_type == kSpherical) { LivoxSpherPoint *p_point_data = (LivoxSpherPoint *) data->data; printf("2\n"); } else if (data->data_type == kExtendCartesian) { LivoxExtendRawPoint *p_point_data = (LivoxExtendRawPoint *) data->data; } else if (data->data_type == kExtendSpherical) { printf("4\n"); LivoxExtendSpherPoint *p_point_data = (LivoxExtendSpherPoint *) data->data; } else if (data->data_type == kDualExtendCartesian) { LivoxDualExtendRawPoint *p_point_data = (LivoxDualExtendRawPoint *) data->data; } else if (data->data_type == kDualExtendSpherical) { LivoxDualExtendSpherPoint *p_point_data = (LivoxDualExtendSpherPoint *) data->data; } else if (data->data_type == kImu) { LivoxImuPoint *p_point_data = (LivoxImuPoint *) data->data; } else if (data->data_type == kTripleExtendCartesian) { LivoxTripleExtendRawPoint *p_point_data = (LivoxTripleExtendRawPoint *) data->data; } else if (data->data_type == kTripleExtendSpherical) { LivoxTripleExtendSpherPoint *p_point_data = (LivoxTripleExtendSpherPoint *) data->data; } } }*/ } void LdsLidar::OnDeviceBroadcast(const BroadcastDeviceInfo *info) { if (info == nullptr) { return; } if (info->dev_type == kDeviceTypeHub) { printf("In lidar mode, couldn't connect a hub : %s\n", info->broadcast_code); return; } /*if(g_snPtr.find(std::string(info->broadcast_code))==false) { printf("sn:%s has not added map size:%d\n",info->broadcast_code,g_snPtr.size()); return; }*/ if(!g_snPtr.find(info->broadcast_code)) { printf(" can not find sn:%s in map\n",info->broadcast_code); return; } livox_status result = kStatusFailure; uint8_t handle = 0; result = AddLidarToConnect(info->broadcast_code, &handle); if (result == kStatusSuccess && handle < kMaxLidarCount) { LdsLidar *lidar = g_snPtr[info->broadcast_code]; SetDataCallback(handle, LdsLidar::GetLidarDataCb, (void *) lidar); LidarDevice *p_lidar = &(lidars_[handle]); p_lidar->handle = handle; p_lidar->connect_state = kConnectStateOff; p_lidar->config.enable_fan = true; p_lidar->config.return_mode = kStrongestReturn; p_lidar->config.coordinate = kCoordinateCartesian; p_lidar->config.imu_rate = kImuFreq200Hz; } else { printf("Add lidar to connect is failed : %d %d \n", result, handle); } } /** Callback function of changing of device state. */ void LdsLidar::OnDeviceChange(const DeviceInfo *info, DeviceEvent type) { if (info == nullptr) { return; } uint8_t handle = info->handle; if (handle >= kMaxLidarCount) { return; } if(g_snPtr.find(info->broadcast_code)==false) return; LdsLidar* ptr=g_snPtr[info->broadcast_code]; LidarDevice *p_lidar = &(lidars_[handle]); if (type == kEventConnect) { QueryDeviceInformation(handle, DeviceInformationCb, ptr); if (p_lidar->connect_state == kConnectStateOff) { p_lidar->connect_state = kConnectStateOn; p_lidar->info = *info; } printf("[WARNING] Lidar sn: [%s] Connect!!!\n", info->broadcast_code); } else if (type == kEventDisconnect) { p_lidar->connect_state = kConnectStateOff; printf("[WARNING] Lidar sn: [%s] Disconnect!!!\n", info->broadcast_code); } else if (type == kEventStateChange) { p_lidar->info = *info; printf("[WARNING] Lidar sn: [%s] StateChange!!!\n", info->broadcast_code); } if (p_lidar->connect_state == kConnectStateOn) { printf("Device Working State %d\n", p_lidar->info.state); if (p_lidar->info.state == kLidarStateInit) { printf("Device State Change Progress %u\n", p_lidar->info.status.progress); } else { printf("Device State Error Code 0X%08x\n", p_lidar->info.status.status_code.error_code); } printf("Device feature %d\n", p_lidar->info.feature); SetErrorMessageCallback(handle, LdsLidar::LidarErrorStatusCb); if (p_lidar->info.state == kLidarStateNormal) { if (p_lidar->config.coordinate != 0) { SetSphericalCoordinate(handle, LdsLidar::SetCoordinateCb, ptr); } else { SetCartesianCoordinate(handle, LdsLidar::SetCoordinateCb, ptr); } p_lidar->config.set_bits |= kConfigCoordinate; if (kDeviceTypeLidarMid40 != info->type) { LidarSetPointCloudReturnMode(handle, (PointCloudReturnMode) (p_lidar->config.return_mode), \ LdsLidar::SetPointCloudReturnModeCb, ptr); p_lidar->config.set_bits |= kConfigReturnMode; } if (kDeviceTypeLidarMid40 != info->type && kDeviceTypeLidarMid70 != info->type) { LidarSetImuPushFrequency(handle, (ImuFreq) (p_lidar->config.imu_rate), \ LdsLidar::SetImuRatePushFrequencyCb, ptr); p_lidar->config.set_bits |= kConfigImuRate; } p_lidar->connect_state = kConnectStateConfig; } } } /** Query the firmware version of Livox LiDAR. */ void LdsLidar::DeviceInformationCb(livox_status status, uint8_t handle, \ DeviceInformationResponse *ack, void *client_data) { if (status != kStatusSuccess) { printf("Device Query Informations Failed : %d\n", status); } if (ack) { printf("firm ver: %d.%d.%d.%d\n", ack->firmware_version[0], ack->firmware_version[1], ack->firmware_version[2], ack->firmware_version[3]); } } /** Callback function of Lidar error message. */ void LdsLidar::LidarErrorStatusCb(livox_status status, uint8_t handle, ErrorMessage *message) { static uint32_t error_message_count = 0; if (message != NULL) { ++error_message_count; if (0 == (error_message_count % 100)) { printf("handle: %u\n", handle); printf("temp_status : %u\n", message->lidar_error_code.temp_status); printf("volt_status : %u\n", message->lidar_error_code.volt_status); printf("motor_status : %u\n", message->lidar_error_code.motor_status); printf("dirty_warn : %u\n", message->lidar_error_code.dirty_warn); printf("firmware_err : %u\n", message->lidar_error_code.firmware_err); printf("pps_status : %u\n", message->lidar_error_code.device_status); printf("fan_status : %u\n", message->lidar_error_code.fan_status); printf("self_heating : %u\n", message->lidar_error_code.self_heating); printf("ptp_status : %u\n", message->lidar_error_code.ptp_status); printf("time_sync_status : %u\n", message->lidar_error_code.time_sync_status); printf("system_status : %u\n", message->lidar_error_code.system_status); } } } void LdsLidar::ControlFanCb(livox_status status, uint8_t handle, \ uint8_t response, void *client_data) { } void LdsLidar::SetPointCloudReturnModeCb(livox_status status, uint8_t handle, \ uint8_t response, void *client_data) { LdsLidar *lds_lidar = static_cast(client_data); if (handle >= kMaxLidarCount) { return; } LidarDevice *p_lidar = &(lidars_[handle]); if (status == kStatusSuccess) { p_lidar->config.set_bits &= ~((uint32_t) (kConfigReturnMode)); printf("Set return mode success!\n"); if (!p_lidar->config.set_bits) { LidarStartSampling(handle, LdsLidar::StartSampleCb, lds_lidar); p_lidar->connect_state = kConnectStateSampling; } } else { LidarSetPointCloudReturnMode(handle, (PointCloudReturnMode) (p_lidar->config.return_mode), \ LdsLidar::SetPointCloudReturnModeCb, lds_lidar); printf("Set return mode fail, try again!\n"); } } void LdsLidar::SetCoordinateCb(livox_status status, uint8_t handle, \ uint8_t response, void *client_data) { LdsLidar *lds_lidar = static_cast(client_data); if (handle >= kMaxLidarCount) { return; } LidarDevice *p_lidar = &(lidars_[handle]); if (status == kStatusSuccess) { p_lidar->config.set_bits &= ~((uint32_t) (kConfigCoordinate)); printf("Set coordinate success!\n"); if (!p_lidar->config.set_bits) { LidarStartSampling(handle, LdsLidar::StartSampleCb, lds_lidar); p_lidar->connect_state = kConnectStateSampling; } } else { if (p_lidar->config.coordinate != 0) { SetSphericalCoordinate(handle, LdsLidar::SetCoordinateCb, lds_lidar); } else { SetCartesianCoordinate(handle, LdsLidar::SetCoordinateCb, lds_lidar); } printf("Set coordinate fail, try again!\n"); } } void LdsLidar::SetImuRatePushFrequencyCb(livox_status status, uint8_t handle, \ uint8_t response, void *client_data) { LdsLidar *lds_lidar = static_cast(client_data); if (handle >= kMaxLidarCount) { return; } LidarDevice *p_lidar = &(lidars_[handle]); if (status == kStatusSuccess) { p_lidar->config.set_bits &= ~((uint32_t) (kConfigImuRate)); printf("Set imu rate success!\n"); if (!p_lidar->config.set_bits) { LidarStartSampling(handle, LdsLidar::StartSampleCb, lds_lidar); p_lidar->connect_state = kConnectStateSampling; } } else { LidarSetImuPushFrequency(handle, (ImuFreq) (p_lidar->config.imu_rate), \ LdsLidar::SetImuRatePushFrequencyCb, lds_lidar); printf("Set imu rate fail, try again!\n"); } } /** Callback function of starting sampling. */ void LdsLidar::StartSampleCb(livox_status status, uint8_t handle, \ uint8_t response, void *client_data) { LdsLidar *lds_lidar = static_cast(client_data); if (handle >= kMaxLidarCount) { return; } LidarDevice *p_lidar = &(lidars_[handle]); if (status == kStatusSuccess) { if (response != 0) { p_lidar->connect_state = kConnectStateOn; printf("Lidar start sample fail : state[%d] handle[%d] res[%d]\n", \ status, handle, response); } else { printf("Lidar start sample success\n"); } } else if (status == kStatusTimeout) { p_lidar->connect_state = kConnectStateOn; printf("Lidar start sample timeout : state[%d] handle[%d] res[%d]\n", \ status, handle, response); } } /** Callback function of stopping sampling. */ void LdsLidar::StopSampleCb(livox_status status, uint8_t handle, \ uint8_t response, void *client_data) { } /** Add broadcast code to whitelist */ int LdsLidar::AddBroadcastCodeToWhitelist(const char *bd_code) { if (!bd_code || (strlen(bd_code) > kBroadcastCodeSize) || \ (whitelist_count_ >= kMaxLidarCount)) { return -1; } if (LdsLidar::FindInWhitelist(bd_code)) { printf("%s is alrealy exist!\n", bd_code); return -1; } strcpy(broadcast_code_whitelist_[whitelist_count_], bd_code); ++whitelist_count_; return 0; } void LdsLidar::AddLocalBroadcastCode(void) { for (size_t i = 0; i < sizeof(local_broadcast_code_list) / sizeof(intptr_t); ++i) { std::string invalid_bd = "000000000"; printf("Local broadcast code : %s\n", local_broadcast_code_list[i]); if ((kBroadcastCodeSize == strlen(local_broadcast_code_list[i]) + 1) && \ (nullptr == strstr(local_broadcast_code_list[i], invalid_bd.c_str()))) { LdsLidar::AddBroadcastCodeToWhitelist(local_broadcast_code_list[i]); } else { printf("Invalid local broadcast code : %s\n", local_broadcast_code_list[i]); } } } bool LdsLidar::FindInWhitelist(const char *bd_code) { if (!bd_code) { return false; } for (uint32_t i = 0; i < whitelist_count_; i++) { if (strncmp(bd_code, broadcast_code_whitelist_[i], kBroadcastCodeSize) == 0) { return true; } } return false; } }