rusefi/firmware/hw_layer/mc33816.cpp

264 lines
6.4 KiB
C++
Raw Normal View History

/*
* @file mc33816.cpp
*
* TL,DR: GDI
*
* The NXP MC33816 is a programmable gate driver IC for precision solenoid control applications.
*
2019-05-08 07:58:43 -07:00
*
2020-07-04 18:38:52 -07:00
* Useful wires:
* 5v, 3(3.3v), Gnd, 12v, VccIO(3v) SPI, DRVEN, RSTB
*
2019-05-08 07:58:43 -07:00
* For MC33816 vs PT2000 differences see
* https://www.nxp.com/docs/en/application-note/AN5203.pdf
*
* @date May 3, 2019
2020-01-07 21:02:40 -08:00
* @author Andrey Belomutskiy, (c) 2012-2020
*/
#include "pch.h"
#if EFI_MC33816
2019-05-03 18:38:34 -07:00
#include "hardware.h"
2019-05-03 18:54:38 -07:00
#include "mpu_util.h"
2024-05-08 18:45:32 -07:00
#include "ignition_controller.h"
2019-05-03 18:38:34 -07:00
static SPIConfig spiCfg = {
.circular = false,
#ifdef _CHIBIOS_RT_CONF_VER_6_1_
.end_cb = NULL,
#else
.slave = false,
.data_cb = NULL,
.error_cb = NULL,
#endif
.ssport = nullptr,
2019-05-03 18:38:34 -07:00
.sspad = 0,
2019-05-03 18:54:38 -07:00
.cr1 =
2024-08-23 20:57:03 -07:00
SPI_CR1_MSTR |
SPI_CR1_SSM | // Software Slave Management, the SSI bit will be internal reference
SPI_CR1_SSI | // Internal Slave Select (active low) set High
SPI_CR1_CPHA |
2024-08-23 20:57:03 -07:00
//SPI_CR1_BR_1 // 5MHz
SPI_CR1_BR_0 | SPI_CR1_BR_1 | SPI_CR1_BR_2 |
2024-08-23 20:57:03 -07:00
SPI_CR1_SPE,
.cr2 = SPI_CR2_SSOE |
SPI_CR2_16BIT_MODE
};
2019-05-03 18:38:34 -07:00
class Pt2001 : public Pt2001Base {
public:
void init();
void initIfNeeded();
2019-05-03 18:38:34 -07:00
protected:
void select() override {
// revive MC33816 driver, also support bus sharing #6781
// should be somewhere but not here spiStart(driver, &spiCfg);
efiPrintf("mc select %s", hwOnChipPhysicalPinName(driver->config->ssport, driver->config->sspad));
spiSelect(driver);
2024-08-23 20:57:03 -07:00
//chris - "no implementation on stm32"
chipSelect.setValue(0);
2020-11-17 22:07:09 -08:00
}
void deselect() override {
spiUnselect(driver);
2024-08-23 20:57:03 -07:00
//chris - "no implementation on stm32"
chipSelect.setValue(1);
}
uint16_t sendRecv(uint16_t tx) override {
return spiPolledExchange(driver, tx);
}
// Send `count` number of 16 bit words from `data`
void sendLarge(const uint16_t* data, size_t count) override {
spiSend(driver, count, data);
}
// GPIO reset and enable pins
void setResetB(bool state) override {
resetB.setValue(state);
}
void setDriveEN(bool state) override {
driven.setValue(state);
}
// GPIO inputs for various pins we need
bool readFlag0() const override {
return efiReadPin(engineConfiguration->mc33816_flag0);
}
// Get battery voltage - only try to init chip when powered
float getVbatt() const override {
// reminder that we do not have sensor subsystem right away
return Sensor::get(SensorType::BatteryVoltage).value_or(0);
}
// CONFIGURATIONS: currents, timings, voltages
float getBoostVoltage() const override {
return engineConfiguration->mc33_hvolt;
}
// Currents in amps
float getBoostCurrent() const override {
return engineConfiguration->mc33_i_boost;
}
float getPeakCurrent() const override {
return engineConfiguration->mc33_i_peak;
}
float getHoldCurrent() const override {
return engineConfiguration->mc33_i_hold;
}
float getPumpPeakCurrent() const override {
return engineConfiguration->mc33_hpfp_i_peak;
}
float getPumpHoldCurrent() const override {
return engineConfiguration->mc33_hpfp_i_hold;
}
// Timings in microseconds
uint16_t getTpeakOff() const override {
return engineConfiguration->mc33_t_peak_off;
}
uint16_t getTpeakTot() const override {
return engineConfiguration->mc33_t_peak_tot;
}
uint16_t getTbypass() const override {
return engineConfiguration->mc33_t_bypass;
}
uint16_t getTholdOff() const override {
return engineConfiguration->mc33_t_hold_off;
}
uint16_t getTHoldTot() const override {
return engineConfiguration->mc33_t_hold_tot;
}
uint16_t getTBoostMin() const override {
return engineConfiguration->mc33_t_min_boost;
}
uint16_t getTBoostMax() const override {
return engineConfiguration->mc33_t_max_boost;
}
uint16_t getPumpTholdOff() const override {
return engineConfiguration->mc33_hpfp_i_hold_off;
}
uint16_t getPumpTholdTot() const override {
return engineConfiguration->mc33_hpfp_max_hold;
}
// Print out an error message
void onError(const char* why) override {
efiPrintf("PT2001 error: %s", why);
}
bool errorOnUnexpectedFlag() override {
efiPrintf("****** unexpected mc33 flag state ******");
return false;
}
void sleepMs(size_t ms) override {
chThdSleepMilliseconds(ms);
}
private:
SPIDriver* driver;
OutputPin resetB;
OutputPin driven;
OutputPin chipSelect;
};
2019-05-03 18:54:38 -07:00
static Pt2001 pt;
void Pt2001::init() {
2020-02-25 19:38:08 -08:00
//
2020-10-19 19:04:06 -07:00
// see setTest33816EngineConfiguration for default configuration
// Pins
if (!isBrainPinValid(engineConfiguration->mc33816_cs) ||
!isBrainPinValid(engineConfiguration->mc33816_rstb) ||
!isBrainPinValid(engineConfiguration->mc33816_driven)) {
return;
2020-10-19 19:04:06 -07:00
}
if (isBrainPinValid(engineConfiguration->mc33816_flag0)) {
efiSetPadMode("mc33816 flag0", engineConfiguration->mc33816_flag0, getInputMode(PI_DEFAULT));
}
2019-05-03 18:38:34 -07:00
2020-04-14 16:23:53 -07:00
chipSelect.initPin("mc33 CS", engineConfiguration->mc33816_cs /*, &engineConfiguration->csPinMode*/);
chipSelect.setValue(1);
2020-04-14 16:23:53 -07:00
// Initialize the chip via ResetB
resetB.initPin("mc33 RESTB", engineConfiguration->mc33816_rstb);
// High Voltage via DRIVEN
driven.initPin("mc33 DRIVEN", engineConfiguration->mc33816_driven);
2024-01-02 20:08:11 -08:00
spiCfg.ssport = getHwPort("mc33816", engineConfiguration->mc33816_cs);
spiCfg.sspad = getHwPin("mc33816", engineConfiguration->mc33816_cs);
// huh, looks like we never set AF for any CS _anywhere_? are we supposed to?
palSetPadMode(spiCfg.ssport, spiCfg.sspad, PAL_MODE_ALTERNATE(5/* << magic for SPI2. todo make visible getSpiAf(driver)*/) | PAL_STM32_OSPEED_HIGHEST);
2019-05-03 18:38:34 -07:00
2024-08-07 20:07:06 -07:00
#if HW_MICRO_RUSEFI
2019-05-03 18:38:34 -07:00
// hard-coded for now, just resolve the conflict with SD card!
engineConfiguration->mc33816spiDevice = SPI_DEVICE_3;
2024-08-07 20:07:06 -07:00
#endif
if (engineConfiguration->mc33816spiDevice == SPI_NONE) {
criticalError("mc33816 needs SPI selection");
}
2019-05-03 18:38:34 -07:00
driver = getSpiDevice(engineConfiguration->mc33816spiDevice);
2024-08-07 20:07:06 -07:00
if (driver == nullptr) {
2019-05-03 18:38:34 -07:00
// error already reported
return;
2019-05-03 18:38:34 -07:00
}
spiStart(driver, &spiCfg);
2024-08-08 10:14:23 -07:00
efiPrintf("mc33 started SPI");
// addConsoleAction("mc33_stats", showStats);
addConsoleAction("mc33_restart", [](){
pt.initIfNeeded();
});
// todo: too soon to read voltage, it has to fail, right?!
initIfNeeded();
2020-04-14 16:23:53 -07:00
}
static bool isInitialized = false;
2019-05-03 18:54:38 -07:00
void Pt2001::initIfNeeded() {
2024-08-23 20:57:03 -07:00
// chris
//if (!isIgnVoltage()) {
// isInitialized = false;
// efiPrintf("unhappy mc33 due to battery voltage");
//} else {
if (!isInitialized) {
isInitialized = restart();
2023-12-12 13:13:36 -08:00
if (isInitialized) {
efiPrintf("happy mc33/PT2001!");
2023-12-12 15:24:42 -08:00
} else {
2023-12-12 18:20:46 -08:00
efiPrintf("unhappy mc33 fault=%d", (int)fault);
2023-12-12 13:13:36 -08:00
}
2020-10-19 19:04:06 -07:00
}
2024-08-23 20:57:03 -07:00
//}
2020-10-19 19:04:06 -07:00
}
void initMc33816() {
pt.init();
}
#endif /* EFI_MC33816 */