Skip to content

Commit

Permalink
VD Interface project for building and testing
Browse files Browse the repository at this point in the history
  • Loading branch information
Tegh25 committed Jan 4, 2025
1 parent bfea139 commit 06bfb49
Show file tree
Hide file tree
Showing 10 changed files with 573 additions and 0 deletions.
20 changes: 20 additions & 0 deletions firmware/projects/VD_Interface/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
# Teghveer Singh Ateliey
# January 4, 2025

# The target executable 'main' is created in the master CMakeLists. Do not change its name.
# We only need to add the source code files and include directories.

# include("${CMAKE_SOURCE_DIR}/cmake/cangen.cmake")

target_sources(main PRIVATE main.cc)

target_include_directories(main
PRIVATE
${CMAKE_CURRENT_SOURCE_DIR}/inc
${CMAKE_CURRENT_SOURCE_DIR}/generated/can
)

message("SOURCES: ${SOURCES}")

# Notice that we don't include any mcal/ subdirectory in this CMake file.
# The master CMakeLists handles platform selection and library linking.
8 changes: 8 additions & 0 deletions firmware/projects/VD_Interface/config.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
canGen:
busses:
- name: veh
node: fc
dbcFile: "../veh.dbc"
- name: pt
node: fc
dbcFile: "../pt.dbc"
237 changes: 237 additions & 0 deletions firmware/projects/VD_Interface/inc/app.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,237 @@
/// @author Blake Freer
/// @date 2024-02-27

#pragma once

#include <cstddef>
#include <cstdint>
#include <memory>
#include <utility>

#include "app.h"
#include "shared/comms/can/can_bus.h"
#include "shared/comms/can/can_msg.h"
#include "shared/periph/adc.h"
#include "shared/periph/can.h"
#include "shared/periph/gpio.h"
#include "shared/util/mappers/linear_map.h"
#include "shared/util/mappers/mapper.h"
#include "shared/util/moving_average.h"
// #include "veh_can_messages.h"

class AnalogInput {
static constexpr size_t kMovingAverageLength = 20;

public:
AnalogInput(shared::periph::ADCInput& adc,
shared::util::Mapper<double, uint16_t>* adc_to_position)
: adc_(adc), adc_to_position_(adc_to_position) {}

double Update() {
uint32_t position = adc_.Read();
moving_average_.LoadValue(uint16_t(position));
return GetPosition();
}

/**
* @brief Get the position from the last `Update()` call.
*/
inline double GetPosition() {
return adc_to_position_->Evaluate(moving_average_.GetValue());
}

private:
shared::periph::ADCInput& adc_;
shared::util::MovingAverage<uint16_t, kMovingAverageLength> moving_average_;
const shared::util::Mapper<double, uint16_t>* adc_to_position_;
};

class Speaker {
public:
Speaker(shared::periph::DigitalOutput& digital_output)
: digital_output_(digital_output) {}

void Update(bool state) const {
digital_output_.Set(state);
}

private:
shared::periph::DigitalOutput& digital_output_;
};

class BrakeLight {
public:
BrakeLight(shared::periph::DigitalOutput& digital_output)
: digital_output_(digital_output) {}

void Update(bool state) const {
digital_output_.Set(state);
}

private:
shared::periph::DigitalOutput& digital_output_;
};

class Button {
public:
Button(shared::periph::DigitalInput& digital_input)
: digital_input_(digital_input) {}

bool Read() const {
return digital_input_.Read();
}

private:
shared::periph::DigitalInput& digital_input_;
};

/**
* @brief Struct containing all data required for each simulink AMK input
* @note (SAM): Do not edit this struct, even though it will be very similar to
* the raw CAN message. Just copy over the CAN message fields to this struct in
* AMKMotor::UpdateInputs, then these values will be passed to the simulink
* input in main.
*/
struct AMKInput {
bool bReserve;
bool bSystemReady;
bool bError;
bool bWarn;
bool bQuitDcOn;
bool bDcOn;
bool bQuitInverterOn;
bool bInverterOn;
bool bDerating;
int16_t ActualVelocity;
int16_t TorqueCurrent;
int16_t MagnetizingCurrent;
int16_t TempMotor;
int16_t TempInverter;
uint16_t ErrorInfo;
int16_t TempIGBT;
};

struct AMKOutput {
uint8_t bInverterOn_tx;
uint8_t bDcOn_tx;
uint8_t bEnable;
uint8_t bErrorReset;
float TargetVelocity;
float TorqueLimitPositiv;
float TorqueLimitNegativ;
};

class AMKMotor {
public:
AMKMotor(shared::can::CanBus& can_bus, uint8_t amk_num)
: can_bus_(can_bus), amk_num_(amk_num) {}

/**
* @brief NOT IMPLEMENTED
* @todo (SAM) construct and send a CAN message from the simulink output
* values in `output`
*/
void Transmit(AMKOutput output) {}

/**
* @brief Read the motor
*
* @param input
*/
AMKInput UpdateInputs() {
/* (SAM) Fill in this section
// You will need to add an instance variable specifying which motor
// this is, so that you can properly
auto amk_can_input = CAN.GetAMKData(this_amk_number_);
*/
return AMKInput{
/* (SAM) populate from the CAN message
.bReserve = amk_can_input.reserve, // sample
.bSystemReady = amk_can_input.system_ready, // sample
.bError = 0,
.bWarn = 0,
.bQuitDcOn = 0,
.bDcOn = 0,
.bQuitInverterOn = 0,
.bInverterOn = 0,
.bDerating = 0,
.ActualVelocity = 0,
.TorqueCurrent = 0,
.MagnetizingCurrent = 0,
.TempMotor = 0,
.TempInverter = 0,
.ErrorInfo = 0,
.TempIGBT = 0,
*/
};
}

private:
shared::can::CanBus& can_bus_;
uint8_t amk_num_ = 0;
};

struct ContactorInput {
bool Pack_Precharge_Feedback;
bool Pack_Negative_Feedback;
bool Pack_Positive_Feedback;
bool HvilFeedback;
int8_t LowThermValue;
int8_t HighThermValue;
int8_t AvgThermValue;
double PackSOC;
};

struct ContactorOutput {
bool prechargeContactorCMD;
bool HVposContactorCMD;
bool HVnegContactorCMD;
};

class Contactors {
public:
Contactors(shared::can::CanBus& can_bus) : can_bus_(can_bus) {}

ContactorInput ReadInput() {
/* (SAM) read from CAN and fill this struct */

// auto msg = can_base_.Read(...);

return ContactorInput{
// (SAM) populate from `msg`
.Pack_Precharge_Feedback = 0, .Pack_Negative_Feedback = 0,
.Pack_Positive_Feedback = 0, .HvilFeedback = 0,
.LowThermValue = 0, .HighThermValue = 0,
.AvgThermValue = 0, .PackSOC = 0,
};
}

// void Transmit(ContactorOutput output) {
// generated::can::ContactorStates contactor_states;

// contactor_states.pack_negative = output.HVnegContactorCMD;
// contactor_states.pack_positive = output.HVposContactorCMD;
// contactor_states.pack_precharge = output.prechargeContactorCMD;

// can_bus_.Send(contactor_states);
// }

private:
shared::can::CanBus& can_bus_;
};

class StatusLight {
public:
StatusLight(shared::periph::DigitalOutput& digital_output)
: digital_output_(digital_output) {}

void Toggle() {
digital_output_.Set(next_value_);
next_value_ = !next_value_;
}

private:
bool next_value_ = true;
shared::periph::DigitalOutput& digital_output_;
};
49 changes: 49 additions & 0 deletions firmware/projects/VD_Interface/inc/simp_vd_interface.cc
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
/// @author Teghveer Singh Ateliey
/// @date 2024-11-23

#include "simp_vd_interface.h"

using namespace ctrl;

SimpVdInterface::SimpVdInterface(float target_slip)
: target_slip(target_slip) {}

VdOutput SimpVdInterface::update(const VdInput& input, int time_ms) {
VdOutput output{
.lm_torque_limit_positive = 0.0f,
.rm_torque_limit_positive = 0.0f,
.lm_torque_limit_negative = 0.0f,
.rm_torque_limit_negative = 0.0f,
.left_motor_speed_request = 1000,
.right_motor_speed_request = 1000,
};

const float pedal_torque_lut_data[][2] = {
{0.0f, 0.0f},
{100.0f, 100.0f}
};

float motor_torque_request = ComputeTorqueRequest(input.driver_torque_request,
input.brake_pedal_postion);
float actual_slip =
CalculateActualSlip(input.wheel_speed_lr, input.wheel_speed_rr,
input.wheel_speed_lf, input.wheel_speed_rf);
float tc_scale_factor = CalculateTCScaleFactor(actual_slip, target_slip, time_ms);

float tv_factor_left;
float tv_factor_right;

float steering_angle = input.tv_enable ? input.steering_angle : 0.0f;
std::tie(tv_factor_left, tv_factor_right) = AdjustTorqueVectoring(
steering_angle, CreateTorqueVectoringFactor(steering_angle));

constexpr int pedal_torque_lut_length = (sizeof(pedal_torque_lut_data)) / (sizeof(pedal_torque_lut_data[0]));
const shared::util::LookupTable<pedal_torque_lut_length> pedal_to_torque{pedal_torque_lut_data};

// Running avg calculation done within CalculateMotorTorque
std::tie(output.lm_torque_limit_positive, output.rm_torque_limit_positive) =
CalculateMotorTorque(pedal_to_torque.Evaluate(motor_torque_request * tc_scale_factor),
tv_factor_right, tv_factor_left);

return output;
}
40 changes: 40 additions & 0 deletions firmware/projects/VD_Interface/inc/simp_vd_interface.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
/// @author Teghveer Singh Ateliey
/// @date 2024-11-23

#pragma once

#include "app.h"
#include "shared/controls/motor_torque.h"
#include "shared/controls/tc_scale_factor.h"
#include "shared/controls/tvFactor.h"
#include "shared/util/mappers/lookup_table.h"

struct VdInput {
float driver_torque_request;
float brake_pedal_postion;
float steering_angle;
float wheel_speed_lr;
float wheel_speed_rr;
float wheel_speed_lf;
float wheel_speed_rf;
bool tv_enable;
};

struct VdOutput {
float lm_torque_limit_positive;
float rm_torque_limit_positive;
float lm_torque_limit_negative;
float rm_torque_limit_negative;
uint16_t left_motor_speed_request;
uint16_t right_motor_speed_request;
};

class SimpVdInterface {
public:
SimpVdInterface(
float target_slip = 0.2f); // default target slip is float 0.2
VdOutput update(const VdInput& input, int time_ms);

private:
float target_slip;
};
Loading

0 comments on commit 06bfb49

Please sign in to comment.