|  |  |  | @ -4,12 +4,16 @@ | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | #include "platform.h" | 
			
		
	
		
			
				
					|  |  |  |  | #include "unit_base.h" | 
			
		
	
		
			
				
					|  |  |  |  | #include "unit_adc.h" | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | #define ADC_INTERNAL | 
			
		
	
		
			
				
					|  |  |  |  | #include "_adc_internal.h" | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | #define DMA_POS(priv) ((priv)->dma_buffer_itemcount - (priv)->DMA_CHx->CNDTR) | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | volatile bool emergency = false; | 
			
		
	
		
			
				
					|  |  |  |  | //#define CRUMB() if(emergency) trap("crumb")
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | static void UADC_JobSendBlockChunk(Job *job) | 
			
		
	
		
			
				
					|  |  |  |  | { | 
			
		
	
		
			
				
					|  |  |  |  |     Unit *unit = job->unit; | 
			
		
	
	
		
			
				
					|  |  |  | @ -21,7 +25,7 @@ static void UADC_JobSendBlockChunk(Job *job) | 
			
		
	
		
			
				
					|  |  |  |  |     uint32_t count = job->data2; | 
			
		
	
		
			
				
					|  |  |  |  |     bool close = (bool) job->data3; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     dbg("Send indices [%d -> %d)", (int)start, (int)(start+count)); | 
			
		
	
		
			
				
					|  |  |  |  | //    dbg("Send indices [%d -> %d)", (int)start, (int)(start+count));
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     TF_TYPE type = close ? EVT_CAPT_DONE : EVT_CAPT_MORE; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
	
		
			
				
					|  |  |  | @ -58,7 +62,7 @@ static void UADC_JobSendTriggerCaptureHeader(Job *job) | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     EventReport_Start(&er); | 
			
		
	
		
			
				
					|  |  |  |  |     priv->stream_frame_id = er.sent_msg_id; | 
			
		
	
		
			
				
					|  |  |  |  |     dbg("Sending TRIG HEADER with id %d (idx %d)", (int)er.sent_msg_id, (int)index_trigd); | 
			
		
	
		
			
				
					|  |  |  |  | //    dbg("Sending TRIG HEADER with id %d (idx %d)", (int)er.sent_msg_id, (int)index_trigd);
 | 
			
		
	
		
			
				
					|  |  |  |  |     { | 
			
		
	
		
			
				
					|  |  |  |  |         // preamble
 | 
			
		
	
		
			
				
					|  |  |  |  |         uint8_t buf[4]; | 
			
		
	
	
		
			
				
					|  |  |  | @ -81,10 +85,10 @@ static void UADC_JobSendTriggerCaptureHeader(Job *job) | 
			
		
	
		
			
				
					|  |  |  |  |                 uint16_t items_from_end = pretrig_remain - index_trigd; | 
			
		
	
		
			
				
					|  |  |  |  |                 assert_param(priv->dma_buffer_itemcount - items_from_end >= index_trigd); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |                 dbg("Pretrig wraparound part: start %d, len %d", | 
			
		
	
		
			
				
					|  |  |  |  |                     (int) (priv->dma_buffer_itemcount - items_from_end), | 
			
		
	
		
			
				
					|  |  |  |  |                     (int) items_from_end | 
			
		
	
		
			
				
					|  |  |  |  |                 ); | 
			
		
	
		
			
				
					|  |  |  |  | //                dbg("Pretrig wraparound part: start %d, len %d",
 | 
			
		
	
		
			
				
					|  |  |  |  | //                    (int) (priv->dma_buffer_itemcount - items_from_end),
 | 
			
		
	
		
			
				
					|  |  |  |  | //                    (int) items_from_end
 | 
			
		
	
		
			
				
					|  |  |  |  | //                );
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |                 EventReport_Data( | 
			
		
	
		
			
				
					|  |  |  |  |                     (uint8_t *) &priv->dma_buffer[priv->dma_buffer_itemcount - | 
			
		
	
	
		
			
				
					|  |  |  | @ -95,10 +99,10 @@ static void UADC_JobSendTriggerCaptureHeader(Job *job) | 
			
		
	
		
			
				
					|  |  |  |  |                 pretrig_remain -= items_from_end; | 
			
		
	
		
			
				
					|  |  |  |  |             } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |             dbg("Pretrig front part: start %d, len %d", | 
			
		
	
		
			
				
					|  |  |  |  |                 (int) (index_trigd - pretrig_remain), | 
			
		
	
		
			
				
					|  |  |  |  |                 (int) pretrig_remain | 
			
		
	
		
			
				
					|  |  |  |  |             ); | 
			
		
	
		
			
				
					|  |  |  |  | //            dbg("Pretrig front part: start %d, len %d",
 | 
			
		
	
		
			
				
					|  |  |  |  | //                (int) (index_trigd - pretrig_remain),
 | 
			
		
	
		
			
				
					|  |  |  |  | //                (int) pretrig_remain
 | 
			
		
	
		
			
				
					|  |  |  |  | //            );
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |             assert_param(pretrig_remain <= index_trigd); | 
			
		
	
		
			
				
					|  |  |  |  |             EventReport_Data((uint8_t *) &priv->dma_buffer[index_trigd - pretrig_remain], | 
			
		
	
	
		
			
				
					|  |  |  | @ -144,6 +148,13 @@ void UADC_DMA_Handler(void *arg) | 
			
		
	
		
			
				
					|  |  |  |  |     struct priv *priv = unit->data; | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(priv); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     if (priv->opmode == ADC_OPMODE_UNINIT) { | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_ClearFlag_HT(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_ClearFlag_TC(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_ClearFlag_TE(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  |         return; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     const uint32_t isrsnapshot = priv->DMAx->ISR; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     if (LL_DMA_IsActiveFlag_G(isrsnapshot, priv->dma_chnum)) { | 
			
		
	
	
		
			
				
					|  |  |  | @ -192,7 +203,13 @@ void UADC_DMA_Handler(void *arg) | 
			
		
	
		
			
				
					|  |  |  |  |                         .data3 = (uint32_t) close, | 
			
		
	
		
			
				
					|  |  |  |  |                         .cb = UADC_JobSendBlockChunk | 
			
		
	
		
			
				
					|  |  |  |  |                     }; | 
			
		
	
		
			
				
					|  |  |  |  |                     scheduleJob(&j); | 
			
		
	
		
			
				
					|  |  |  |  |                     if (!scheduleJob(&j)) { | 
			
		
	
		
			
				
					|  |  |  |  |                         // Abort if we can't queue - the stream would tear and we'd hog the system with error messages
 | 
			
		
	
		
			
				
					|  |  |  |  |                         dbg("(!) Buffers overflow, abort capture"); | 
			
		
	
		
			
				
					|  |  |  |  |                         emergency = true; | 
			
		
	
		
			
				
					|  |  |  |  |                         UADC_SwitchMode(unit, ADC_OPMODE_EMERGENCY_SHUTDOWN); | 
			
		
	
		
			
				
					|  |  |  |  |                         return; | 
			
		
	
		
			
				
					|  |  |  |  |                     } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |                     if (close) { | 
			
		
	
		
			
				
					|  |  |  |  | //                        dbg("End of capture");
 | 
			
		
	
	
		
			
				
					|  |  |  | @ -243,10 +260,11 @@ void UADC_ADC_EOS_Handler(void *arg) | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(priv); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     LL_ADC_ClearFlag_EOS(priv->ADCx); | 
			
		
	
		
			
				
					|  |  |  |  |     if (priv->opmode == ADC_OPMODE_UNINIT) return; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     // Wait for the DMA to complete copying the last sample
 | 
			
		
	
		
			
				
					|  |  |  |  |     uint16_t dmapos; | 
			
		
	
		
			
				
					|  |  |  |  |     while ((dmapos = (uint16_t) DMA_POS(priv)) % priv->nb_channels != 0); | 
			
		
	
		
			
				
					|  |  |  |  |     hw_wait_while((dmapos = (uint16_t) DMA_POS(priv)) % priv->nb_channels != 0, 100); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     uint32_t sample_pos; | 
			
		
	
		
			
				
					|  |  |  |  |     if (dmapos == 0) { | 
			
		
	
	
		
			
				
					|  |  |  | @ -313,6 +331,7 @@ void UADC_HandleTrigger(Unit *unit, uint8_t edge_type, uint64_t timestamp) | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(unit); | 
			
		
	
		
			
				
					|  |  |  |  |     struct priv *priv = unit->data; | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(priv); | 
			
		
	
		
			
				
					|  |  |  |  |     if (priv->opmode == ADC_OPMODE_UNINIT) return; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     if (priv->trig_holdoff != 0 && priv->trig_holdoff_remain > 0) { | 
			
		
	
		
			
				
					|  |  |  |  | //        dbg("Trig discarded due to holdoff.");
 | 
			
		
	
	
		
			
				
					|  |  |  | @ -323,7 +342,7 @@ void UADC_HandleTrigger(Unit *unit, uint8_t edge_type, uint64_t timestamp) | 
			
		
	
		
			
				
					|  |  |  |  |         priv->trig_holdoff_remain = priv->trig_holdoff; | 
			
		
	
		
			
				
					|  |  |  |  |         // Start the tick
 | 
			
		
	
		
			
				
					|  |  |  |  |         unit->tick_interval = 1; | 
			
		
	
		
			
				
					|  |  |  |  |         unit->_tick_cnt = 0; | 
			
		
	
		
			
				
					|  |  |  |  |         unit->_tick_cnt = 1; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     priv->stream_startpos = (uint16_t) DMA_POS(priv); | 
			
		
	
	
		
			
				
					|  |  |  | @ -349,6 +368,7 @@ void UADC_StartBlockCapture(Unit *unit, uint32_t len, TF_ID frame_id) | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(unit); | 
			
		
	
		
			
				
					|  |  |  |  |     struct priv *priv = unit->data; | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(priv); | 
			
		
	
		
			
				
					|  |  |  |  |     if (priv->opmode == ADC_OPMODE_UNINIT) return; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     priv->stream_frame_id = frame_id; | 
			
		
	
		
			
				
					|  |  |  |  |     priv->stream_startpos = (uint16_t) DMA_POS(priv); | 
			
		
	
	
		
			
				
					|  |  |  | @ -363,11 +383,11 @@ void UADC_StartStream(Unit *unit, TF_ID frame_id) | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(unit); | 
			
		
	
		
			
				
					|  |  |  |  |     struct priv *priv = unit->data; | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(priv); | 
			
		
	
		
			
				
					|  |  |  |  |     if (priv->opmode == ADC_OPMODE_UNINIT) return; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     priv->stream_frame_id = frame_id; | 
			
		
	
		
			
				
					|  |  |  |  |     priv->stream_startpos = (uint16_t) DMA_POS(priv); | 
			
		
	
		
			
				
					|  |  |  |  |     priv->stream_serial = 0; | 
			
		
	
		
			
				
					|  |  |  |  | //    dbg("Start streaming.");
 | 
			
		
	
		
			
				
					|  |  |  |  |     UADC_SwitchMode(unit, ADC_OPMODE_STREAM); | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
	
		
			
				
					|  |  |  | @ -377,8 +397,8 @@ void UADC_StopStream(Unit *unit) | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(unit); | 
			
		
	
		
			
				
					|  |  |  |  |     struct priv *priv = unit->data; | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(priv); | 
			
		
	
		
			
				
					|  |  |  |  |     if (priv->opmode == ADC_OPMODE_UNINIT) return; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | //    dbg("Stop stream.");
 | 
			
		
	
		
			
				
					|  |  |  |  |     UADC_ReportEndOfStream(unit); | 
			
		
	
		
			
				
					|  |  |  |  |     UADC_SwitchMode(unit, ADC_OPMODE_IDLE); | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
	
		
			
				
					|  |  |  | @ -390,6 +410,16 @@ void UADC_updateTick(Unit *unit) | 
			
		
	
		
			
				
					|  |  |  |  |     struct priv *priv = unit->data; | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(priv); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     // Recover from shutdown after a delay
 | 
			
		
	
		
			
				
					|  |  |  |  |     if (priv->opmode == ADC_OPMODE_EMERGENCY_SHUTDOWN) { | 
			
		
	
		
			
				
					|  |  |  |  |         dbg("Recovering from emergency shutdown"); | 
			
		
	
		
			
				
					|  |  |  |  |         UADC_SwitchMode(unit, ADC_OPMODE_IDLE); | 
			
		
	
		
			
				
					|  |  |  |  |         LL_TIM_EnableCounter(priv->TIMx); | 
			
		
	
		
			
				
					|  |  |  |  |         UADC_ReportEndOfStream(unit); | 
			
		
	
		
			
				
					|  |  |  |  |         unit->tick_interval = 0; | 
			
		
	
		
			
				
					|  |  |  |  |         return; | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     if (priv->trig_holdoff_remain > 0) { | 
			
		
	
		
			
				
					|  |  |  |  |         priv->trig_holdoff_remain--; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
	
		
			
				
					|  |  |  | @ -406,10 +436,14 @@ void UADC_SwitchMode(Unit *unit, enum uadc_opmode new_mode) | 
			
		
	
		
			
				
					|  |  |  |  |     struct priv *priv = unit->data; | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param(priv); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     if (new_mode == priv->opmode) return; // nothing to do
 | 
			
		
	
		
			
				
					|  |  |  |  |     const enum uadc_opmode old_mode = priv->opmode; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     if (new_mode == old_mode) return; // nothing to do
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     // if un-itied, can go only to IDLE
 | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param((priv->opmode != ADC_OPMODE_UNINIT) || (new_mode == ADC_OPMODE_IDLE)); | 
			
		
	
		
			
				
					|  |  |  |  |     assert_param((old_mode != ADC_OPMODE_UNINIT) || (new_mode == ADC_OPMODE_IDLE)); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     priv->opmode = ADC_OPMODE_UNINIT; | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     if (new_mode == ADC_OPMODE_UNINIT) { | 
			
		
	
		
			
				
					|  |  |  |  | //        dbg("ADC switch -> UNINIT");
 | 
			
		
	
	
		
			
				
					|  |  |  | @ -436,11 +470,12 @@ void UADC_SwitchMode(Unit *unit, enum uadc_opmode new_mode) | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_DisableIT_TC(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     else if (new_mode == ADC_OPMODE_IDLE || new_mode == ADC_OPMODE_REARM_PENDING) { | 
			
		
	
		
			
				
					|  |  |  |  | //        dbg("ADC switch -> IDLE or IDLE/REARM_PENDING");
 | 
			
		
	
		
			
				
					|  |  |  |  |         // IDLE and ARMED are identical with the exception that the trigger condition is not checked
 | 
			
		
	
		
			
				
					|  |  |  |  |         // ARMED can be only entered from IDLE, thus we do the init only here.
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         // In IDLE, we don't need the DMA interrupts
 | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_ClearFlag_HT(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_ClearFlag_TC(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_DisableIT_HT(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_DisableIT_TC(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
	
		
			
				
					|  |  |  | @ -448,7 +483,7 @@ void UADC_SwitchMode(Unit *unit, enum uadc_opmode new_mode) | 
			
		
	
		
			
				
					|  |  |  |  |         LL_ADC_ClearFlag_EOS(priv->ADCx); | 
			
		
	
		
			
				
					|  |  |  |  |         LL_ADC_EnableIT_EOS(priv->ADCx); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         if (priv->opmode == ADC_OPMODE_UNINIT) { | 
			
		
	
		
			
				
					|  |  |  |  |         if (old_mode == ADC_OPMODE_UNINIT) { | 
			
		
	
		
			
				
					|  |  |  |  |             // Nothing is started yet - this is the only way to leave UNINIT
 | 
			
		
	
		
			
				
					|  |  |  |  |             LL_ADC_Enable(priv->ADCx); | 
			
		
	
		
			
				
					|  |  |  |  |             LL_DMA_EnableChannel(priv->DMAx, priv->dma_chnum); | 
			
		
	
	
		
			
				
					|  |  |  | @ -457,9 +492,28 @@ void UADC_SwitchMode(Unit *unit, enum uadc_opmode new_mode) | 
			
		
	
		
			
				
					|  |  |  |  |             LL_ADC_REG_StartConversion(priv->ADCx); | 
			
		
	
		
			
				
					|  |  |  |  |         } | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     else if (new_mode == ADC_OPMODE_EMERGENCY_SHUTDOWN) { | 
			
		
	
		
			
				
					|  |  |  |  |         // Emergency shutdown is used when the job queue overflows and the stream is torn
 | 
			
		
	
		
			
				
					|  |  |  |  |         // This however doesn't help in the case when user sets such a high frequency
 | 
			
		
	
		
			
				
					|  |  |  |  |         // that the whole app becomes unresponsive due to the completion ISR, need to verify the value manually.
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_ClearFlag_HT(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_ClearFlag_TC(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_DisableIT_HT(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_DisableIT_TC(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         LL_TIM_DisableCounter(priv->TIMx); | 
			
		
	
		
			
				
					|  |  |  |  |         UADC_SetSampleRate(unit, 10000); // fallback to a known safe value
 | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         LL_ADC_ClearFlag_EOS(priv->ADCx); | 
			
		
	
		
			
				
					|  |  |  |  |         LL_ADC_DisableIT_EOS(priv->ADCx); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         unit->tick_interval = 0; | 
			
		
	
		
			
				
					|  |  |  |  |         unit->_tick_cnt = 250; // 1-off
 | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  |     else if (new_mode == ADC_OPMODE_ARMED) { | 
			
		
	
		
			
				
					|  |  |  |  | //        dbg("ADC switch -> ARMED");
 | 
			
		
	
		
			
				
					|  |  |  |  |         assert_param(priv->opmode == ADC_OPMODE_IDLE || priv->opmode == ADC_OPMODE_REARM_PENDING); | 
			
		
	
		
			
				
					|  |  |  |  |         assert_param(old_mode == ADC_OPMODE_IDLE || old_mode == ADC_OPMODE_REARM_PENDING); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         // avoid firing immediately by the value jumping across the scale
 | 
			
		
	
		
			
				
					|  |  |  |  |         priv->trig_prev_level = priv->last_samples[priv->trigger_source]; | 
			
		
	
	
		
			
				
					|  |  |  | @ -469,7 +523,7 @@ void UADC_SwitchMode(Unit *unit, enum uadc_opmode new_mode) | 
			
		
	
		
			
				
					|  |  |  |  |         new_mode == ADC_OPMODE_BLCAP) { | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  | //        dbg("ADC switch -> TRIG'D / STREAM / BLOCK");
 | 
			
		
	
		
			
				
					|  |  |  |  |         assert_param(priv->opmode == ADC_OPMODE_ARMED || priv->opmode == ADC_OPMODE_IDLE); | 
			
		
	
		
			
				
					|  |  |  |  |         assert_param(old_mode == ADC_OPMODE_ARMED || old_mode == ADC_OPMODE_IDLE); | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |         // during the capture, we disallow direct readout and averaging to reduce overhead
 | 
			
		
	
		
			
				
					|  |  |  |  |         LL_ADC_DisableIT_EOS(priv->ADCx); | 
			
		
	
	
		
			
				
					|  |  |  | @ -484,6 +538,6 @@ void UADC_SwitchMode(Unit *unit, enum uadc_opmode new_mode) | 
			
		
	
		
			
				
					|  |  |  |  |         LL_DMA_EnableIT_TC(priv->DMAx, priv->dma_chnum); | 
			
		
	
		
			
				
					|  |  |  |  |     } | 
			
		
	
		
			
				
					|  |  |  |  | 
 | 
			
		
	
		
			
				
					|  |  |  |  |     // the actual switch
 | 
			
		
	
		
			
				
					|  |  |  |  |     dbg("Now setting the new opmode"); | 
			
		
	
		
			
				
					|  |  |  |  |     priv->opmode = new_mode; | 
			
		
	
		
			
				
					|  |  |  |  | } | 
			
		
	
	
		
			
				
					|  |  |  | 
 |