wip direct, unfinished config

remotes/github/direct
Ondřej Hruška 6 years ago
parent 20dfa7e158
commit 75efa12338
Signed by: MightyPork
GPG Key ID: 2C5FD5035250423D
  1. 30
      platform/irq_dispatcher.c
  2. 16
      platform/timebase.c
  3. 66
      units/fcap/_fcap_core.c
  4. 13
      units/fcap/_fcap_init.c
  5. 15
      units/fcap/_fcap_internal.h
  6. 37
      units/fcap/unit_fcap.c

@ -64,7 +64,9 @@ static struct callbacks_ {
struct cbslot tim2; struct cbslot tim2;
struct cbslot tim6; struct cbslot tim6;
struct cbslot tim7; struct cbslot tim7;
struct cbslot tim14;
struct cbslot tim15; struct cbslot tim15;
struct cbslot tim16;
struct cbslot adc1; struct cbslot adc1;
@ -104,23 +106,31 @@ void irqd_init(void)
// NVIC_EnableIRQ(TIM1_IRQn); /*!< TIM1 global Interrupt */ // NVIC_EnableIRQ(TIM1_IRQn); /*!< TIM1 global Interrupt */
NVIC_EnableIRQ(TIM2_IRQn); /*!< TIM2 global Interrupt */ NVIC_EnableIRQ(TIM2_IRQn); /*!< TIM2 global Interrupt */
HAL_NVIC_SetPriority(TIM2_IRQn, 2, 0); // Used by FCAP
// NVIC_EnableIRQ(TIM3_IRQn); /*!< TIM3 global Interrupt */ // NVIC_EnableIRQ(TIM3_IRQn); /*!< TIM3 global Interrupt */
NVIC_EnableIRQ(TIM6_DAC_IRQn); /*!< TIM6 global and DAC channel underrun error Interrupt */ NVIC_EnableIRQ(TIM6_DAC_IRQn); /*!< TIM6 global and DAC channel underrun error Interrupt */
HAL_NVIC_SetPriority(TIM7_IRQn, 2, 0); // Used for DAC timing HAL_NVIC_SetPriority(TIM7_IRQn, 2, 0); // Used for DAC timing
NVIC_EnableIRQ(TIM7_IRQn); /*!< TIM7 global Interrupt */ NVIC_EnableIRQ(TIM7_IRQn); /*!< TIM7 global Interrupt */
HAL_NVIC_SetPriority(TIM7_IRQn, 2, 0); HAL_NVIC_SetPriority(TIM7_IRQn, 2, 0);// this will be for dac (?)
/* Tim14 is used for HAL timebase, because SysTick is used to time FreeRTOS and has the lowest priority. */ NVIC_EnableIRQ(TIM14_IRQn); /*used by fcap as a time reference for direct capture */ /*!< TIM14 global Interrupt */
/* Tim14's priority is set to 0 in the init routine, which runs early in the startup sequence */ HAL_NVIC_SetPriority(TIM14_IRQn, 2, 0);
// NVIC_EnableIRQ(TIM14_IRQn); /*!< TIM14 global Interrupt */
NVIC_EnableIRQ(TIM15_IRQn); /*!< TIM15 global Interrupt */ NVIC_EnableIRQ(TIM15_IRQn); /*!< TIM15 global Interrupt */
HAL_NVIC_SetPriority(TIM15_IRQn, 2, 0); HAL_NVIC_SetPriority(TIM15_IRQn, 2, 0); // Used by ADC
NVIC_EnableIRQ(TIM16_IRQn); /*!< TIM16 global Interrupt */
HAL_NVIC_SetPriority(TIM16_IRQn, 2, 0);
// NVIC_EnableIRQ(TIM16_IRQn); /*!< TIM16 global Interrupt */ /* Tim17 is used for HAL timebase, because SysTick is used to time FreeRTOS and has the lowest priority. */
/* Tim17's priority is set to 0 in the init routine, which runs early in the startup sequence */
// NVIC_EnableIRQ(TIM17_IRQn); /*!< TIM17 global Interrupt */ // NVIC_EnableIRQ(TIM17_IRQn); /*!< TIM17 global Interrupt */
// NVIC_EnableIRQ(I2C1_IRQn); /*!< I2C1 Event Interrupt & EXTI Line23 Interrupt (I2C1 wakeup) */ // NVIC_EnableIRQ(I2C1_IRQn); /*!< I2C1 Event Interrupt & EXTI Line23 Interrupt (I2C1 wakeup) */
// NVIC_EnableIRQ(I2C2_IRQn); /*!< I2C2 Event Interrupt */ // NVIC_EnableIRQ(I2C2_IRQn); /*!< I2C2 Event Interrupt */
// NVIC_EnableIRQ(SPI1_IRQn); /*!< SPI1 global Interrupt */ // NVIC_EnableIRQ(SPI1_IRQn); /*!< SPI1 global Interrupt */
@ -163,7 +173,10 @@ static struct cbslot *get_slot_for_periph(void *periph)
else if (periph == TIM2) slot = &callbacks.tim2; else if (periph == TIM2) slot = &callbacks.tim2;
else if (periph == TIM6) slot = &callbacks.tim6; else if (periph == TIM6) slot = &callbacks.tim6;
else if (periph == TIM7) slot = &callbacks.tim7; else if (periph == TIM7) slot = &callbacks.tim7;
else if (periph == TIM14) slot = &callbacks.tim14;
else if (periph == TIM15) slot = &callbacks.tim15; else if (periph == TIM15) slot = &callbacks.tim15;
else if (periph == TIM16) slot = &callbacks.tim16;
// 17 - used by timebase
else if (periph == ADC1) slot = &callbacks.adc1; else if (periph == ADC1) slot = &callbacks.adc1;
@ -326,6 +339,11 @@ void TIM15_IRQHandler(void)
CALL_IRQ_HANDLER(callbacks.tim15); CALL_IRQ_HANDLER(callbacks.tim15);
} }
void TIM16_IRQHandler(void)
{
CALL_IRQ_HANDLER(callbacks.tim16);
}
void ADC1_COMP_IRQHandler(void) void ADC1_COMP_IRQHandler(void)
{ {
CALL_IRQ_HANDLER(callbacks.adc1); CALL_IRQ_HANDLER(callbacks.adc1);

@ -7,10 +7,12 @@
// ---------------------------- HAL TIMEBASE ----------------------------- // ---------------------------- HAL TIMEBASE -----------------------------
#define TIMEBASE_TIMER TIM14 #define TIMEBASE_TIMER TIM17
HAL_StatusTypeDef HAL_InitTick(uint32_t TickPriority) HAL_StatusTypeDef HAL_InitTick(uint32_t TickPriority)
{ {
// EDIT - used 17 instead because 14 was needed for fcap
// TIM14 is a simple 16-bit timer timer with no special features. // TIM14 is a simple 16-bit timer timer with no special features.
// This makes it a good choice for the timebase generation. We set it to generate // This makes it a good choice for the timebase generation. We set it to generate
// an interrupt every 1 ms // an interrupt every 1 ms
@ -19,9 +21,9 @@ HAL_StatusTypeDef HAL_InitTick(uint32_t TickPriority)
// - TIM14 is always up-counting // - TIM14 is always up-counting
// - using APB1 clock // - using APB1 clock
__HAL_RCC_TIM14_CLK_ENABLE(); __HAL_RCC_TIM17_CLK_ENABLE();
NVIC_SetPriority(TIM14_IRQn, TickPriority); // highest possible priority NVIC_SetPriority(TIM17_IRQn, TickPriority); // highest possible priority
NVIC_EnableIRQ(TIM14_IRQn); NVIC_EnableIRQ(TIM17_IRQn);
/* Compute TIM1 clock */ /* Compute TIM1 clock */
uint32_t uwTimclock = HAL_RCC_GetPCLK1Freq(); uint32_t uwTimclock = HAL_RCC_GetPCLK1Freq();
@ -45,7 +47,7 @@ HAL_StatusTypeDef HAL_InitTick(uint32_t TickPriority)
static volatile uint32_t uwUptimeMs = 0; static volatile uint32_t uwUptimeMs = 0;
/* TIMEBASE TIMER ISR */ /* TIMEBASE TIMER ISR */
void TIM14_IRQHandler(void) void TIM17_IRQHandler(void)
{ {
uwUptimeMs++; uwUptimeMs++;
LL_TIM_ClearFlag_UPDATE(TIMEBASE_TIMER); LL_TIM_ClearFlag_UPDATE(TIMEBASE_TIMER);
@ -89,10 +91,10 @@ uint64_t PTIM_GetMicrotime(void)
uwMicros = TIMEBASE_TIMER->CNT; uwMicros = TIMEBASE_TIMER->CNT;
uwMillis = uwUptimeMs; uwMillis = uwUptimeMs;
if (LL_TIM_IsActiveFlag_UPDATE(TIM14)) { if (LL_TIM_IsActiveFlag_UPDATE(TIMEBASE_TIMER)) {
// This means the timer has overflown after we disabled IRQ // This means the timer has overflown after we disabled IRQ
// Use the last CNT value before the overflow // Use the last CNT value before the overflow
uwMicros = TIM14->ARR; // this is 999us uwMicros = TIMEBASE_TIMER->ARR; // this is 999us
} }
} }
vPortExitCritical(); vPortExitCritical();

@ -7,6 +7,10 @@
#define FCAP_INTERNAL #define FCAP_INTERNAL
#include "_fcap_internal.h" #include "_fcap_internal.h"
void UFCAP_StopMeasurement(Unit *unit);
void UFCAP_ConfigureForPWMCapture(Unit *unit);
void UFCAP_ConfigureForDirectCapture(Unit *unit);
static void UFCAP_PWMBurstReportJob(Job *job) static void UFCAP_PWMBurstReportJob(Job *job)
{ {
Unit *unit = job->unit; Unit *unit = job->unit;
@ -39,7 +43,6 @@ void UFCAP_TimerHandler(void *arg)
if (priv->opmode == OPMODE_PWM_CONT) { if (priv->opmode == OPMODE_PWM_CONT) {
if (LL_TIM_IsActiveFlag_CC1(TIMx)) { if (LL_TIM_IsActiveFlag_CC1(TIMx)) {
// assert_param(!LL_TIM_IsActiveFlag_CC1OVR(TIMx));
if (priv->n_skip > 0) { if (priv->n_skip > 0) {
priv->n_skip--; priv->n_skip--;
} else { } else {
@ -51,7 +54,6 @@ void UFCAP_TimerHandler(void *arg)
} }
if (LL_TIM_IsActiveFlag_CC2(TIMx)) { if (LL_TIM_IsActiveFlag_CC2(TIMx)) {
// assert_param(!LL_TIM_IsActiveFlag_CC2OVR(TIMx));
priv->pwm_cont.ontime = LL_TIM_IC_GetCaptureCH2(TIMx); priv->pwm_cont.ontime = LL_TIM_IC_GetCaptureCH2(TIMx);
LL_TIM_ClearFlag_CC2(TIMx); LL_TIM_ClearFlag_CC2(TIMx);
LL_TIM_ClearFlag_CC2OVR(TIMx); LL_TIM_ClearFlag_CC2OVR(TIMx);
@ -59,7 +61,6 @@ void UFCAP_TimerHandler(void *arg)
} }
else if (priv->opmode == OPMODE_PWM_BURST) { else if (priv->opmode == OPMODE_PWM_BURST) {
if (LL_TIM_IsActiveFlag_CC1(TIMx)) { if (LL_TIM_IsActiveFlag_CC1(TIMx)) {
// assert_param(!LL_TIM_IsActiveFlag_CC1OVR(TIMx));
const uint32_t period = LL_TIM_IC_GetCaptureCH1(TIMx); const uint32_t period = LL_TIM_IC_GetCaptureCH1(TIMx);
const uint32_t ontime = priv->pwm_burst.ontime; const uint32_t ontime = priv->pwm_burst.ontime;
@ -85,7 +86,6 @@ void UFCAP_TimerHandler(void *arg)
} }
if (LL_TIM_IsActiveFlag_CC2(TIMx)) { if (LL_TIM_IsActiveFlag_CC2(TIMx)) {
// assert_param(!LL_TIM_IsActiveFlag_CC2OVR(TIMx));
priv->pwm_burst.ontime = LL_TIM_IC_GetCaptureCH2(TIMx); priv->pwm_burst.ontime = LL_TIM_IC_GetCaptureCH2(TIMx);
LL_TIM_ClearFlag_CC2(TIMx); LL_TIM_ClearFlag_CC2(TIMx);
LL_TIM_ClearFlag_CC2OVR(TIMx); LL_TIM_ClearFlag_CC2OVR(TIMx);
@ -95,6 +95,9 @@ void UFCAP_TimerHandler(void *arg)
// clear everything - in idle it would cycle in the handler forever // clear everything - in idle it would cycle in the handler forever
TIMx->SR = 0; TIMx->SR = 0;
} }
else {
trap("Unhandled fcap TIMx irq");
}
} }
static void UFCAP_ClearTimerConfig(Unit *unit) static void UFCAP_ClearTimerConfig(Unit *unit)
@ -120,9 +123,9 @@ static void UFCAP_ClearTimerConfig(Unit *unit)
void UFCAP_StopMeasurement(Unit *unit) void UFCAP_StopMeasurement(Unit *unit)
{ {
struct priv * const priv = unit->data; struct priv * const priv = unit->data;
TIM_TypeDef * const TIMx = priv->TIMx;
LL_TIM_DeInit(TIMx); // clear all flags and settings LL_TIM_DeInit(priv->TIMx); // clear all flags and settings
LL_TIM_DeInit(priv->TIMy); // clear all flags and settings
} }
/** /**
@ -161,6 +164,11 @@ void UFCAP_SwitchMode(Unit *unit, enum fcap_opmode opmode)
priv->n_skip = 1; // discard the first cycle (will be incomplete) priv->n_skip = 1; // discard the first cycle (will be incomplete)
UFCAP_ConfigureForPWMCapture(unit); // is also stopped and restarted UFCAP_ConfigureForPWMCapture(unit); // is also stopped and restarted
break; break;
case OPMODE_COUNTER_CONT:
UFCAP_ConfigureForDirectCapture(unit);
break;
default: default:
trap("Unhandled opmode %d", (int)opmode); trap("Unhandled opmode %d", (int)opmode);
} }
@ -208,3 +216,49 @@ void UFCAP_ConfigureForPWMCapture(Unit *unit)
LL_TIM_EnableIT_CC2(TIMx); LL_TIM_EnableIT_CC2(TIMx);
LL_TIM_EnableCounter(TIMx); LL_TIM_EnableCounter(TIMx);
} }
/**
* Configure peripherals for an indirect capture (PWM measurement) - continuous or burst
* @param unit
*/
void UFCAP_ConfigureForDirectCapture(Unit *unit)
{
struct priv * const priv = unit->data;
const uint32_t ll_ch_a = priv->ll_ch_a; //
UFCAP_ClearTimerConfig(unit);
{
TIM_TypeDef *const TIMy = priv->TIMy;
uint16_t presc = PLAT_AHB_MHZ * 500; // this produces 2 kHz
uint32_t count = 2001;
LL_TIM_SetPrescaler(TIMy, (uint32_t) (presc - 1));
LL_TIM_SetAutoReload(TIMy, count - 1);
LL_TIM_EnableARRPreload(TIMy);
LL_TIM_GenerateEvent_UPDATE(TIMy);
LL_TIM_SetOnePulseMode(TIMy, LL_TIM_ONEPULSEMODE_SINGLE); // TODO check if this works
LL_TIM_OC_EnableFast(TIMy, LL_TIM_CHANNEL_CH1);
dbg("TIMy presc %d, count %d", (int) presc, (int) count);
LL_TIM_SetTriggerOutput(TIMy, LL_TIM_TRGO_OC1REF);
LL_TIM_OC_SetMode(TIMy, LL_TIM_CHANNEL_CH1, LL_TIM_OCMODE_PWM1); // 1 until CC, then 0
LL_TIM_OC_SetCompareCH1(TIMy, count - 1); // XXX maybe this must be lower
LL_TIM_CC_EnableChannel(TIMy, LL_TIM_CHANNEL_CH1); // enable the output channel that produces a trigger
}
{
// TIMx - the slave
TIM_TypeDef *const TIMx = priv->TIMx;
LL_TIM_SetSlaveMode(TIMx, LL_TIM_SLAVEMODE_GATED);
LL_TIM_SetTriggerInput(TIMx, LL_TIM_TS_ITR3); // ITR3 is TIM14 which we use as TIMy
LL_TIM_EnableMasterSlaveMode(TIMx);
LL_TIM_EnableExternalClock(TIMx); // TODO must check and deny this mode if the pin is not on CH1 = external trigger input
LL_TIM_EnableCounter(TIMx);
}
LL_TIM_EnableCounter(priv->TIMy); // XXX this will start the pulse (maybe)
}

@ -33,6 +33,9 @@ error_t UFCAP_init(Unit *unit)
TIM_TypeDef * const TIMx = TIM2; TIM_TypeDef * const TIMx = TIM2;
Resource timRsc = R_TIM2; Resource timRsc = R_TIM2;
TIM_TypeDef * const TIMy = TIM14;
Resource tim2Rsc = R_TIM14;
uint32_t ll_ch_a = 0; uint32_t ll_ch_a = 0;
uint32_t ll_ch_b = 0; uint32_t ll_ch_b = 0;
@ -79,6 +82,7 @@ error_t UFCAP_init(Unit *unit)
TRY(rsc_claim_pin(unit, priv->signal_pname, priv->signal_pnum)); TRY(rsc_claim_pin(unit, priv->signal_pname, priv->signal_pnum));
TRY(rsc_claim(unit, timRsc)); TRY(rsc_claim(unit, timRsc));
TRY(rsc_claim(unit, tim2Rsc));
// ---- INIT ---- // ---- INIT ----
assert_param(ll_ch_a != ll_ch_b); assert_param(ll_ch_a != ll_ch_b);
@ -91,7 +95,9 @@ error_t UFCAP_init(Unit *unit)
TRY(hw_configure_gpio_af(priv->signal_pname, priv->signal_pnum, ll_timpin_af)); TRY(hw_configure_gpio_af(priv->signal_pname, priv->signal_pnum, ll_timpin_af));
hw_periph_clock_enable(TIMx); hw_periph_clock_enable(TIMx);
hw_periph_clock_enable(TIMy);
irqd_attach(TIMx, UFCAP_TimerHandler, unit); irqd_attach(TIMx, UFCAP_TimerHandler, unit);
// TODO attach TIMy to a handler
UFCAP_SwitchMode(unit, OPMODE_IDLE); UFCAP_SwitchMode(unit, OPMODE_IDLE);
@ -108,8 +114,13 @@ void UFCAP_deInit(Unit *unit)
UFCAP_SwitchMode(unit, OPMODE_IDLE); UFCAP_SwitchMode(unit, OPMODE_IDLE);
TIM_TypeDef *TIMx = priv->TIMx; TIM_TypeDef *TIMx = priv->TIMx;
TIM_TypeDef *TIMy = priv->TIMy;
LL_TIM_DeInit(TIMx); LL_TIM_DeInit(TIMx);
irqd_attach(TIMx, UFCAP_TimerHandler, unit); LL_TIM_DeInit(TIMy);
irqd_detach(TIMx, UFCAP_TimerHandler);
// TODO detach TIMy when any handler is added
hw_periph_clock_disable(TIMx);
hw_periph_clock_disable(TIMy);
} }
// Release all resources, deinit pins // Release all resources, deinit pins

@ -16,6 +16,8 @@ enum fcap_opmode {
OPMODE_BUSY = 1, // used after capture is done, before it's reported OPMODE_BUSY = 1, // used after capture is done, before it's reported
OPMODE_PWM_CONT = 2, OPMODE_PWM_CONT = 2,
OPMODE_PWM_BURST = 3, // averaging OPMODE_PWM_BURST = 3, // averaging
OPMODE_COUNTER_CONT = 4,
OPMODE_COUNTER_BURST = 5, // averaging
}; };
/** Private data structure */ /** Private data structure */
@ -26,6 +28,7 @@ struct priv {
// internal state // internal state
TIM_TypeDef *TIMx; TIM_TypeDef *TIMx;
TIM_TypeDef *TIMy; // used as a timebase source for TIMx in direct mode
uint32_t ll_ch_b; uint32_t ll_ch_b;
uint32_t ll_ch_a; uint32_t ll_ch_a;
bool a_direct; bool a_direct;
@ -49,6 +52,10 @@ struct priv {
uint16_t n_count; //!< Periods captured uint16_t n_count; //!< Periods captured
uint16_t n_target; //!< Periods captured - requested count uint16_t n_target; //!< Periods captured - requested count
} pwm_burst; } pwm_burst;
struct {
uint32_t last_count; //!< Pulse count in the last capture window
} cnt_cont;
}; };
}; };
@ -79,12 +86,8 @@ void UFCAP_deInit(Unit *unit);
// ------------------------------------------------------------------------ // ------------------------------------------------------------------------
void UFCAP_TimerHandler(void *arg);
void UFCAP_StopMeasurement(Unit *unit);
void UFCAP_ConfigureForPWMCapture(Unit *unit);
void UFCAP_SwitchMode(Unit *unit, enum fcap_opmode opmode); void UFCAP_SwitchMode(Unit *unit, enum fcap_opmode opmode);
void UFCAP_TimerHandler(void *arg);
#endif //GEX_F072_FCAP_INTERNAL_H #endif //GEX_F072_FCAP_INTERNAL_H

@ -12,29 +12,32 @@
enum FcapCmd_ { enum FcapCmd_ {
CMD_STOP = 0, CMD_STOP = 0,
CMD_PWM_CONT_START = 1, CMD_INDIRECT_CONT_START = 1,
CMD_PWM_BURST_START = 2, CMD_INDIRECT_BURST_START = 2,
CMD_PWM_CONT_READ = 10, CMD_DIRECT_CONT_START = 3,
CMD_INDIRECT_CONT_READ = 10,
CMD_DIRECT_CONT_READ = 11,
}; };
/** Handle a request message */ /** Handle a request message */
static error_t UFCAP_handleRequest(Unit *unit, TF_ID frame_id, uint8_t command, PayloadParser *pp) static error_t UFCAP_handleRequest(Unit *unit, TF_ID frame_id, uint8_t command, PayloadParser *pp)
{ {
struct priv *priv = unit->data; struct priv *priv = unit->data;
PayloadBuilder pb = pb_start(unit_tmp512, UNIT_TMP_LEN, NULL);
switch (command) { switch (command) {
case CMD_STOP: case CMD_STOP:
UFCAP_SwitchMode(unit, OPMODE_IDLE); UFCAP_SwitchMode(unit, OPMODE_IDLE);
return E_SUCCESS; return E_SUCCESS;
case CMD_PWM_CONT_START: case CMD_INDIRECT_CONT_START:
if (priv->opmode == OPMODE_PWM_CONT) return E_SUCCESS; // no-op if (priv->opmode == OPMODE_PWM_CONT) return E_SUCCESS; // no-op
if (priv->opmode != OPMODE_IDLE) return E_BUSY; if (priv->opmode != OPMODE_IDLE) return E_BUSY;
UFCAP_SwitchMode(unit, OPMODE_PWM_CONT); UFCAP_SwitchMode(unit, OPMODE_PWM_CONT);
return E_SUCCESS; return E_SUCCESS;
case CMD_PWM_BURST_START: case CMD_INDIRECT_BURST_START:
if (priv->opmode != OPMODE_IDLE) return E_BAD_MODE; if (priv->opmode != OPMODE_IDLE) return E_BAD_MODE;
uint16_t count = pp_u16(pp); uint16_t count = pp_u16(pp);
@ -43,7 +46,7 @@ static error_t UFCAP_handleRequest(Unit *unit, TF_ID frame_id, uint8_t command,
UFCAP_SwitchMode(unit, OPMODE_PWM_BURST); UFCAP_SwitchMode(unit, OPMODE_PWM_BURST);
return E_SUCCESS; return E_SUCCESS;
case CMD_PWM_CONT_READ: case CMD_INDIRECT_CONT_READ:
if (priv->opmode != OPMODE_PWM_CONT) { if (priv->opmode != OPMODE_PWM_CONT) {
return E_BAD_MODE; return E_BAD_MODE;
} }
@ -51,7 +54,6 @@ static error_t UFCAP_handleRequest(Unit *unit, TF_ID frame_id, uint8_t command,
return E_BUSY; return E_BUSY;
} }
PayloadBuilder pb = pb_start(unit_tmp512, UNIT_TMP_LEN, NULL);
pb_u16(&pb, PLAT_AHB_MHZ); pb_u16(&pb, PLAT_AHB_MHZ);
pb_u32(&pb, priv->pwm_cont.last_period); pb_u32(&pb, priv->pwm_cont.last_period);
pb_u32(&pb, priv->pwm_cont.last_ontime); pb_u32(&pb, priv->pwm_cont.last_ontime);
@ -59,6 +61,27 @@ static error_t UFCAP_handleRequest(Unit *unit, TF_ID frame_id, uint8_t command,
com_respond_pb(frame_id, MSG_SUCCESS, &pb); com_respond_pb(frame_id, MSG_SUCCESS, &pb);
return E_SUCCESS; return E_SUCCESS;
case CMD_DIRECT_CONT_START:
if (priv->opmode == OPMODE_COUNTER_CONT) return E_SUCCESS; // no-op
if (priv->opmode != OPMODE_IDLE) return E_BUSY;
UFCAP_SwitchMode(unit, OPMODE_COUNTER_CONT);
return E_SUCCESS;
case CMD_DIRECT_CONT_READ:
if (priv->opmode != OPMODE_COUNTER_CONT) {
return E_BAD_MODE;
}
if (priv->cnt_cont.last_count == 0) {
return E_BUSY;
}
// TODO also add the window len - may need to shorten it for fast signals (or if fast capture is required)
pb_u32(&pb, priv->cnt_cont.last_count);
com_respond_pb(frame_id, MSG_SUCCESS, &pb);
return E_SUCCESS;
default: default:
return E_UNKNOWN_COMMAND; return E_UNKNOWN_COMMAND;
} }

Loading…
Cancel
Save