Skip to content

Commit

Permalink
Merge remote-tracking branch 'origin' into inverse-kinematics-requests
Browse files Browse the repository at this point in the history
  • Loading branch information
Geeoon committed Sep 29, 2023
2 parents 82e7b1b + 29ac65c commit 5adaa35
Show file tree
Hide file tree
Showing 20 changed files with 119 additions and 50 deletions.
1 change: 1 addition & 0 deletions .clang-format
Original file line number Diff line number Diff line change
Expand Up @@ -17,3 +17,4 @@ IncludeCategories:
Priority: 3
IndentCaseLabels: true
PointerAlignment: Left
AlwaysBreakTemplateDeclarations: Yes
2 changes: 1 addition & 1 deletion src/CAN/CAN.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@
#include <sys/types.h>

extern "C" {
#include "../HindsightCAN/CANCommon.h"
#include <HindsightCAN/CANCommon.h>
}

using robot::types::DataPoint;
Expand Down
6 changes: 3 additions & 3 deletions src/CAN/CANMotor.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -8,9 +8,9 @@
#include <thread>

extern "C" {
#include "../HindsightCAN/CANCommon.h"
#include "../HindsightCAN/CANMotorUnit.h"
#include "../HindsightCAN/CANPacket.h"
#include <HindsightCAN/CANCommon.h>
#include <HindsightCAN/CANMotorUnit.h>
#include <HindsightCAN/CANPacket.h>
}

using namespace std::chrono_literals;
Expand Down
6 changes: 3 additions & 3 deletions src/CAN/CANMotor_AVR.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -6,9 +6,9 @@
#include <thread>

extern "C" {
#include "../HindsightCAN/CANCommon.h"
#include "../HindsightCAN/CANMotorUnit.h"
#include "../HindsightCAN/CANPacket.h"
#include <HindsightCAN/CANCommon.h>
#include <HindsightCAN/CANMotorUnit.h>
#include <HindsightCAN/CANPacket.h>
}

using namespace std::chrono_literals;
Expand Down
2 changes: 1 addition & 1 deletion src/CAN/CANUtils.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@
#include <sstream>

extern "C" {
#include "../HindsightCAN/CANPacket.h"
#include <HindsightCAN/CANPacket.h>
}

namespace can {
Expand Down
2 changes: 1 addition & 1 deletion src/CAN/FakeCANBoard.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@
#include <vector>

extern "C" {
#include "../HindsightCAN/CANScience.h"
#include <HindsightCAN/CANScience.h>
}

using namespace std::chrono_literals;
Expand Down
4 changes: 2 additions & 2 deletions src/CAN/TestIMU.cpp
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
#include "../log.h"
#include "CAN.h"
extern "C" {
#include "../HindsightCAN/CANCommon.h"
#include "../HindsightCAN/CANSerialNumbers.h"
#include <HindsightCAN/CANCommon.h>
#include <HindsightCAN/CANSerialNumbers.h>
}

#include <chrono>
Expand Down
11 changes: 9 additions & 2 deletions src/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@ set(CMAKE_CXX_STANDARD_REQUIRED ON)

project(Rover LANGUAGES CXX C)

include(FetchContent)

# If a build type was not explicitly specified, default to "RelWithDebInfo" (Same optimization
# as the "Release" configuration but with debugging symbols).
#
Expand Down Expand Up @@ -109,7 +111,12 @@ find_package(nlohmann_json 3.2.0 REQUIRED)

# Find the CAN library; contains packet and protocol definitions and does not
# actually require physical CAN to be present.
find_package(HindsightCAN 1.4.0 REQUIRED)
FetchContent_Declare(
HindsightCAN
GIT_REPOSITORY https://github.com/huskyroboticsteam/HindsightCAN.git
GIT_TAG 7397787fbb04687bf03c8fbac9a1db56f245fc9a
)
FetchContent_MakeAvailable(HindsightCAN)

find_package(Threads REQUIRED)
find_package(OpenCV REQUIRED)
Expand Down Expand Up @@ -197,7 +204,7 @@ else()
CAN/CANStubs.cpp)
endif()
target_link_libraries(can_interface PUBLIC
HindsightCAN::HindsightCAN
HindsightCAN
Threads::Threads
utils
)
Expand Down
2 changes: 1 addition & 1 deletion src/Rover.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -132,8 +132,8 @@ void parseCommandLine(int argc, char** argv) {
int main(int argc, char** argv) {
parseCommandLine(argc, argv);
Globals::AUTONOMOUS = false;
robot::world_interface_init();
Globals::websocketServer.start();
robot::world_interface_init();
auto mcProto = std::make_unique<net::mc::MissionControlProtocol>(Globals::websocketServer);
Globals::websocketServer.addProtocol(std::move(mcProto));
rospub::init();
Expand Down
8 changes: 2 additions & 6 deletions src/ardupilot/ArduPilotInterface.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -25,11 +25,7 @@ DataPoint<gpscoords_t> readGPSCoords() {
} // namespace gps

namespace robot {
DataPoint<double> readIMUHeading() {
auto imu_heading = ardupilot_protocol->getIMU();
if (imu_heading.isValid()) {
return DataPoint<double>(imu_heading.getTime(), imu_heading.getData().yaw);
}
return DataPoint<double>();
DataPoint<eulerangles_t> readIMU() {
return ardupilot_protocol->getIMU();
}
} // namespace robot
2 changes: 1 addition & 1 deletion src/ardupilot/ArduPilotProtocol.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -65,7 +65,7 @@ bool ArduPilotProtocol::validateGPSRequest(const json& j) {

void ArduPilotProtocol::handleGPSRequest(const json& j) {
double lat = j["lat"];
double lon = j["lat"];
double lon = j["lon"];
{
std::lock_guard<std::mutex> lock(_lastGPSMutex);
_lastGPS = gpscoords_t{lat, lon};
Expand Down
41 changes: 25 additions & 16 deletions src/network/MissionControlProtocol.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
#include "../Globals.h"
#include "../base64/base64_img.h"
#include "../log.h"
#include "../utils/core.h"
#include "../utils/json.h"
#include "../world_interface/data.h"
#include "../world_interface/world_interface.h"
Expand Down Expand Up @@ -185,24 +186,32 @@ void MissionControlProtocol::handleJointPowerRequest(const json& j) {
}

void MissionControlProtocol::sendRoverPos() {
auto heading = robot::readIMUHeading();
auto gps = robot::readGPS();
if (gps.isValid() && heading.isValid()) {
double orientX = 0.0;
double orientY = 0.0;
double orientZ = std::sin(heading.getData() / 2);
double orientW = std::cos(heading.getData() / 2);
double posX = gps.getData()[0];
double posY = gps.getData()[1];
auto imu = robot::readIMU();
auto gps = gps::readGPSCoords();
log(LOG_DEBUG, "imu_valid=%d, gps_valid=%d\n", util::to_string(imu.isValid()),
util::to_string(gps.isValid()));
if (imu.isValid()) {
auto rpy = imu.getData();
Eigen::Quaterniond quat(Eigen::AngleAxisd(rpy.roll, Eigen::Vector3d::UnitX()) *
Eigen::AngleAxisd(rpy.pitch, Eigen::Vector3d::UnitY()) *
Eigen::AngleAxisd(rpy.yaw, Eigen::Vector3d::UnitZ()));
double posX = 0, posY = 0;
if (gps.isValid()) {
posX = gps.getData().lon;
posY = gps.getData().lat;
}
double posZ = 0.0;
double gpsRecency = util::durationToSec(dataclock::now() - gps.getTime());
double headingRecency = util::durationToSec(dataclock::now() - heading.getTime());
double recency = std::max(gpsRecency, headingRecency);
double imuRecency = util::durationToSec(dataclock::now() - imu.getTime());
double recency = imuRecency;
if (gps.isValid()) {
double gpsRecency = util::durationToSec(dataclock::now() - gps.getTime());
recency = std::max(recency, gpsRecency);
}
json msg = {{"type", ROVER_POS_REP_TYPE},
{"orientW", orientW},
{"orientX", orientX},
{"orientY", orientY},
{"orientZ", orientZ},
{"orientW", quat.w()},
{"orientX", quat.x()},
{"orientY", quat.y()},
{"orientZ", quat.z()},
{"posX", posX},
{"posY", posY},
{"posZ", posZ},
Expand Down
5 changes: 5 additions & 0 deletions src/utils/core.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,11 @@ bool almostEqual(double a, double b, double threshold) {
return std::abs(a - b) < threshold;
}

template <>
std::string to_string<bool>(const bool& val) {
return val ? "true" : "false";
}

frozen::string freezeStr(const std::string& str) {
return frozen::string(str.c_str(), str.size());
}
Expand Down
12 changes: 11 additions & 1 deletion src/utils/core.h
Original file line number Diff line number Diff line change
Expand Up @@ -44,10 +44,20 @@ std::unordered_set<K> keySet(const std::unordered_map<K, V>& input) {
* @return A string representation of that value, as a std::string. The exact representation of
* the value is up to the implementation.
*/
template <typename T> std::string to_string(const T& val) {
template <typename T>
std::string to_string(const T& val) {
return std::to_string(val);
}

/**
* @brief Converts a boolean to a string.
*
* @param val The value to get the string representation of.
* @return "true" iff val, otherwise "false".
*/
template <>
std::string to_string<bool>(const bool& val);

/**
* @brief Convert the given std::string to a frozen::string.
*/
Expand Down
25 changes: 23 additions & 2 deletions src/world_interface/data.h
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
#include <bitset>
#include <chrono>
#include <optional>
#include <type_traits>
#include <vector>

#include <frozen/string.h>
Expand All @@ -29,7 +30,8 @@ using dataclock = std::chrono::steady_clock;
/** @brief A point in time as measured by dataclock */
using datatime_t = std::chrono::time_point<dataclock>;

template <typename T> class DataPoint;
template <typename T>
class DataPoint;

/**
* @brief A data structure that represents when each landmark was seen.
Expand Down Expand Up @@ -108,7 +110,8 @@ constexpr auto name_to_jointid = frozen::make_unordered_map<frozen::string, join
*
* @tparam T The type of data measured from the sensor.
*/
template <typename T> class DataPoint {
template <typename T>
class DataPoint {
public:
/**
* @brief Construct an invalid DataPoint, holding no data
Expand Down Expand Up @@ -200,6 +203,24 @@ template <typename T> class DataPoint {
return isValid() ? getData() : defaultData;
}

/**
* @brief Transforms the data in this datapoint by the given function.
*
* @tparam F A callable type that accepts @p T and outputs some other type.
* @param f The function that transforms data.
* @return DataPoint<std::invoke_result_t<F, T>> A new datapoint that holds
* the output of `f(getData())` and has the same timestamp as this datapoint
* if it's valid, otherwise an empty datapoint.
*/
template <typename F>
DataPoint<std::invoke_result_t<F, T>> transform(const F& f) {
if (isValid()) {
return DataPoint<std::invoke_result_t<F, T>>(getTime(), f(getData()));
} else {
return {};
}
}

private:
/**
* @brief The time and measurement data.
Expand Down
4 changes: 4 additions & 0 deletions src/world_interface/gps_common_interface.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,10 @@ static std::optional<GPSToMetersConverter> converter;

namespace robot {

DataPoint<double> readIMUHeading() {
return readIMU().transform([](const navtypes::eulerangles_t& e) { return e.yaw; });
}

DataPoint<point_t> readGPS() {
DataPoint<gpscoords_t> coords = gps::readGPSCoords();
if (!coords) {
Expand Down
2 changes: 1 addition & 1 deletion src/world_interface/noop_world.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@ landmarks_t readLandmarks() {
return landmarks_t{};
}

DataPoint<double> readIMUHeading() {
DataPoint<navtypes::eulerangles_t> readIMU() {
return {};
}

Expand Down
21 changes: 13 additions & 8 deletions src/world_interface/simulator_interface.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -39,10 +39,11 @@ const std::map<motorid_t, std::string> motorNameMap = {
{motorid_t::elbow, "elbow"},
{motorid_t::forearm, "forearm"},
{motorid_t::wrist, "wrist"},
{motorid_t::hand, "hand"}};
{motorid_t::hand, "hand"},
{motorid_t::activeSuspension, "activeSuspension"}};

DataPoint<double> lastHeading;
std::mutex headingMutex;
DataPoint<navtypes::eulerangles_t> lastAttitude;
std::mutex attitudeMutex;

DataPoint<gpscoords_t> lastGPS;
std::mutex gpsMutex;
Expand Down Expand Up @@ -149,8 +150,12 @@ void handleIMU(json msg) {
double qw = msg["w"];

double heading = util::quatToHeading(qw, qx, qy, qz);
std::lock_guard<std::mutex> guard(headingMutex);
lastHeading = {heading};
Eigen::Quaterniond q(qw, qx, qy, qz);
q.normalize();
Eigen::Vector3d euler = q.toRotationMatrix().eulerAngles(0, 1, 2);
std::lock_guard<std::mutex> guard(attitudeMutex);
lastAttitude =
navtypes::eulerangles_t{.roll = euler(0), .pitch = euler(1), .yaw = euler(2)};
}

void handleCamFrame(json msg) {
Expand Down Expand Up @@ -364,9 +369,9 @@ DataPoint<pose_t> readVisualOdomVel() {
return DataPoint<pose_t>{};
}

DataPoint<double> readIMUHeading() {
std::lock_guard<std::mutex> guard(headingMutex);
return lastHeading;
DataPoint<navtypes::eulerangles_t> readIMU() {
std::lock_guard<std::mutex> guard(attitudeMutex);
return lastAttitude;
}

DataPoint<pose_t> getTruePose() {
Expand Down
11 changes: 11 additions & 0 deletions src/world_interface/world_interface.h
Original file line number Diff line number Diff line change
Expand Up @@ -165,6 +165,17 @@ types::DataPoint<navtypes::point_t> readGPS();
*/
types::DataPoint<double> readIMUHeading();

/**
* @brief Read the rover attitude from the IMU.
*
* Euler angles are in radians, in RPY format (i.e. XYZ extrinsic)
*
* @see https://en.wikipedia.org/wiki/Euler_angles#Tait%E2%80%93Bryan_angles
*
* @return types::DataPoint<navtypes::eulerangles_t>
*/
types::DataPoint<navtypes::eulerangles_t> readIMU();

/**
* @brief Get the ground truth pose, if available. This is only available in simulation.
*
Expand Down
2 changes: 1 addition & 1 deletion update_deps.sh
Original file line number Diff line number Diff line change
Expand Up @@ -4,5 +4,5 @@ sudo apt update
sudo apt install -y build-essential unzip gnupg2 lsb-release git \
cmake libeigen3-dev libopencv-dev libopencv-contrib-dev \
libwebsocketpp-dev libboost-system-dev gpsd gpsd-clients libgps-dev nlohmann-json3-dev \
catch2 urg-lidar rplidar ublox-linux hindsight-can=1.4.0 frozen libargparse-dev libavutil-dev \
catch2 urg-lidar rplidar ublox-linux frozen libargparse-dev libavutil-dev \
libx264-dev

0 comments on commit 5adaa35

Please sign in to comment.