libxr  1.0
Want to be the best embedded framework
Loading...
Searching...
No Matches
stm32_adc.cpp
1#include "stm32_adc.hpp"
2
3#include <cstddef>
4#include <limits>
5
6#include "libxr_def.hpp"
7#include "stm32_dcache.hpp"
8
9#ifdef HAL_ADC_MODULE_ENABLED
10
11using namespace LibXR;
12
13using H = ADC_HandleTypeDef*;
14
15#if defined(HAL_ADC_MODULE_ENABLED) && !defined(ADC_CALIB_OFFSET_AND_LINEARITY) && \
16 !defined(ADC_CALIB_OFFSET_LINEARITY) && !defined(ADC_CALIB_OFFSET) && \
17 !defined(ADC_SINGLE_ENDED)
18extern "C" HAL_StatusTypeDef
19 __attribute__((weak)) HAL_ADCEx_Calibration_Start(ADC_HandleTypeDef* hadc)
20{
21 (void)hadc;
22 return HAL_OK;
23}
24#endif
25
26namespace
27{
28
29uint8_t CalculateFilterSize(RawData dma_buff, std::size_t channel_count)
30{
31 ASSERT(channel_count > 0U);
32
33 const std::size_t filter_size = dma_buff.size_ / channel_count / sizeof(uint16_t);
34 ASSERT(filter_size > 0U);
35 ASSERT(filter_size <= std::numeric_limits<uint8_t>::max());
36 return static_cast<uint8_t>(filter_size);
37}
38
39} // namespace
40
41STM32ADC::Channel::Channel(STM32ADC* adc, uint8_t index, uint32_t ch)
42 : adc_(adc), index_(index), ch_(ch)
43{
44}
45
46float STM32ADC::Channel::Read() { return adc_->ReadChannel(index_); }
47
48STM32ADC::STM32ADC(ADC_HandleTypeDef* hadc, RawData dma_buff,
49 std::initializer_list<uint32_t> channels, float vref)
50 : hadc_(hadc),
51 NUM_CHANNELS(channels.size()),
52 filter_size_(CalculateFilterSize(dma_buff, channels.size())),
53 use_dma_(hadc_->DMA_Handle != nullptr),
54 dma_buffer_(dma_buff),
55 resolution_(GetADCResolution<ADC_HandleTypeDef>{}.Get(hadc)),
56 channels_(new Channel*[NUM_CHANNELS]),
57 vref_(vref)
58{
59 auto it = channels.begin();
60 for (uint8_t i = 0; i < NUM_CHANNELS; ++i)
61 {
62 channels_[i] = new Channel(this, i, *it++);
63 }
64
65#if defined(ADC_CALIB_OFFSET_AND_LINEARITY) && defined(ADC_SINGLE_ENDED)
66 HAL_ADCEx_Calibration_Start(hadc, ADC_CALIB_OFFSET_AND_LINEARITY, ADC_SINGLE_ENDED);
67#elif defined(ADC_CALIB_OFFSET_LINEARITY) && defined(ADC_SINGLE_ENDED)
68 HAL_ADCEx_Calibration_Start(hadc, ADC_CALIB_OFFSET_LINEARITY, ADC_SINGLE_ENDED);
69#elif defined(ADC_CALIB_OFFSET) && defined(ADC_SINGLE_ENDED)
70 HAL_ADCEx_Calibration_Start(hadc, ADC_CALIB_OFFSET, ADC_SINGLE_ENDED);
71#elif defined(ADC_SINGLE_ENDED)
72 HAL_ADCEx_Calibration_Start(hadc, ADC_SINGLE_ENDED);
73#else
74 HAL_ADCEx_Calibration_Start(hadc);
75#endif
76
77 if (use_dma_)
78 {
79 /* DMA must be in circular mode */
80 AssertContinuousConvModeEnabled<H>(hadc_);
81 AssertDMAContReqEnabled<H>(hadc_);
82 AssertDMACircular<H>(hadc_);
83 AssertNbrOfConvEq<H>(hadc_, NUM_CHANNELS);
84 HAL_ADC_Start_DMA(hadc_, reinterpret_cast<uint32_t*>(dma_buffer_.addr_),
85 NUM_CHANNELS * filter_size_);
86 }
87 else
88 {
89 AssertNbrOfConvEq<H>(hadc_, 1);
90 AssertContinuousConvModeDisabled<H>(hadc_);
91 HAL_ADC_Start(hadc_);
92 }
93}
94
96{
97 use_dma_ ? HAL_ADC_Stop_DMA(hadc_) : HAL_ADC_Stop(hadc_);
98 for (uint8_t i = 0; i < NUM_CHANNELS; ++i)
99 {
100 delete channels_[i];
101 }
102 delete[] channels_;
103}
104
105STM32ADC::Channel& STM32ADC::GetChannel(uint8_t index) { return *channels_[index]; }
106
107float STM32ADC::ReadChannel(uint8_t channel)
108{
109 if (channel >= NUM_CHANNELS)
110 {
111 ASSERT(false);
112 return -1.0f;
113 }
114
115 uint16_t* buffer = reinterpret_cast<uint16_t*>(dma_buffer_.addr_);
116 if (use_dma_)
117 {
118 STM32_InvalidateDCacheByAddr(buffer, filter_size_ * NUM_CHANNELS * 2);
119 uint32_t sum = 0;
120 for (uint8_t i = 0; i < filter_size_; ++i)
121 {
122 sum += buffer[channel + i * NUM_CHANNELS];
123 }
124 return ConvertToVoltage(static_cast<float>(sum) / static_cast<float>(filter_size_));
125 }
126
127 ADC_ChannelConfTypeDef config = {};
128 uint32_t time = 0;
129 UNUSED(time);
130#if defined(ADC_SAMPLETIME_16CYCLES)
131 time = ADC_SAMPLETIME_16CYCLES;
132#elif defined(ADC_SAMPLETIME_16CYCLES_5)
133 time = ADC_SAMPLETIME_16CYCLES_5;
134#elif defined(ADC_SAMPLETIME_17CYCLES)
135 time = ADC_SAMPLETIME_17CYCLES;
136#elif defined(ADC_SAMPLETIME_17CYCLES_5)
137 time = ADC_SAMPLETIME_17CYCLES_5;
138#elif defined(ADC_SAMPLETIME_18CYCLES)
139 time = ADC_SAMPLETIME_18CYCLES;
140#elif defined(ADC_SAMPLETIME_18CYCLES_5)
141 time = ADC_SAMPLETIME_18CYCLES_5;
142#elif defined(ADC_SAMPLETIME_19CYCLES)
143 time = ADC_SAMPLETIME_19CYCLES;
144#elif defined(ADC_SAMPLETIME_19CYCLES_5)
145 time = ADC_SAMPLETIME_19CYCLES_5;
146#elif defined(ADC_SAMPLETIME_20CYCLES)
147 time = ADC_SAMPLETIME_20CYCLES;
148#elif defined(ADC_SAMPLETIME_20CYCLES_5)
149 time = ADC_SAMPLETIME_20CYCLES_5;
150#elif defined(ADC_SAMPLETIME_21CYCLES)
151 time = ADC_SAMPLETIME_21CYCLES;
152#elif defined(ADC_SAMPLETIME_21CYCLES_5)
153 time = ADC_SAMPLETIME_21CYCLES_5;
154#elif defined(ADC_SAMPLETIME_22CYCLES)
155 time = ADC_SAMPLETIME_22CYCLES;
156#elif defined(ADC_SAMPLETIME_22CYCLES_5)
157 time = ADC_SAMPLETIME_22CYCLES_5;
158#elif defined(ADC_SAMPLETIME_23CYCLES)
159 time = ADC_SAMPLETIME_23CYCLES;
160#elif defined(ADC_SAMPLETIME_23CYCLES_5)
161 time = ADC_SAMPLETIME_23CYCLES_5;
162#elif defined(ADC_SAMPLETIME_24CYCLES)
163 time = ADC_SAMPLETIME_24CYCLES;
164#elif defined(ADC_SAMPLETIME_24CYCLES_5)
165 time = ADC_SAMPLETIME_24CYCLES_5;
166#elif defined(ADC_SAMPLETIME_25CYCLES)
167 time = ADC_SAMPLETIME_25CYCLES;
168#elif defined(ADC_SAMPLETIME_25CYCLES_5)
169 time = ADC_SAMPLETIME_25CYCLES_5;
170#elif defined(ADC_SAMPLETIME_26CYCLES)
171 time = ADC_SAMPLETIME_26CYCLES;
172#elif defined(ADC_SAMPLETIME_26CYCLES_5)
173 time = ADC_SAMPLETIME_26CYCLES_5;
174#elif defined(ADC_SAMPLETIME_27CYCLES)
175 time = ADC_SAMPLETIME_27CYCLES;
176#elif defined(ADC_SAMPLETIME_27CYCLES_5)
177 time = ADC_SAMPLETIME_27CYCLES_5;
178#elif defined(ADC_SAMPLETIME_28CYCLES)
179 time = ADC_SAMPLETIME_28CYCLES;
180#elif defined(ADC_SAMPLETIME_28CYCLES_5)
181 time = ADC_SAMPLETIME_28CYCLES_5;
182#elif defined(ADC_SAMPLETIME_29CYCLES)
183 time = ADC_SAMPLETIME_29CYCLES;
184#elif defined(ADC_SAMPLETIME_29CYCLES_5)
185 time = ADC_SAMPLETIME_29CYCLES_5;
186#elif defined(ADC_SAMPLETIME_30CYCLES)
187 time = ADC_SAMPLETIME_30CYCLES;
188#elif defined(ADC_SAMPLETIME_30CYCLES_5)
189 time = ADC_SAMPLETIME_30CYCLES_5;
190#elif defined(ADC_SAMPLETIME_31CYCLES)
191 time = ADC_SAMPLETIME_31CYCLES;
192#elif defined(ADC_SAMPLETIME_31CYCLES_5)
193 time = ADC_SAMPLETIME_31CYCLES_5;
194#elif defined(ADC_SAMPLETIME_32CYCLES)
195 time = ADC_SAMPLETIME_32CYCLES;
196#elif defined(ADC_SAMPLETIME_32CYCLES_5)
197 time = ADC_SAMPLETIME_32CYCLES_5;
198#error "Unsupported sample time"
199#endif
200
201 config.Channel = channels_[channel]->ch_;
202#if defined(ADC_REGULAR_RANK_1)
203 config.Rank = ADC_REGULAR_RANK_1;
204#else
205 config.Rank = 1;
206#endif
207#if defined(ADC_SINGLE_ENDED) && !defined(STM32L0)
208 config.SingleDiff = ADC_SINGLE_ENDED;
209#endif
210#if defined(ADC_OFFSET_NONE)
211 config.OffsetNumber = ADC_OFFSET_NONE;
212 config.Offset = 0;
213#endif
214#if !defined(STM32L0)
215 config.SamplingTime = time;
216#endif
217
218 uint32_t expected = 0U;
219
220 if (!locked_.compare_exchange_strong(expected, 0xF0F0F0F0U, std::memory_order_acquire,
221 std::memory_order_relaxed))
222 {
223 // Multiple threads are working on the same adc peripheral
224 // Please use dma mode
225 ASSERT(false);
226 return 0.0f;
227 }
228
229 HAL_ADC_ConfigChannel(hadc_, &config);
230
231 uint32_t sum = 0;
232 for (uint8_t i = 0; i < filter_size_; ++i)
233 {
234 HAL_ADC_Start(hadc_);
235 HAL_ADC_PollForConversion(hadc_, 20);
236 buffer[channel + i * NUM_CHANNELS] = HAL_ADC_GetValue(hadc_);
237 sum += buffer[channel + i * NUM_CHANNELS];
238 }
239
240 locked_.store(0, std::memory_order_release);
241
242 return ConvertToVoltage(static_cast<float>(sum) / static_cast<float>(filter_size_));
243}
244
245float STM32ADC::ConvertToVoltage(float adc_value)
246{
247 return adc_value * vref_ / resolution_;
248}
249
250#endif
可写原始数据视图 / Mutable raw data view
size_t size_
数据字节数 / Data size in bytes
void * addr_
数据起始地址 / Data start address
ADC 通道对象 / ADC channel object.
float Read() override
读取 ADC 电压值 / Read ADC voltage
Definition stm32_adc.cpp:46
STM32 ADC 驱动实现 / STM32 ADC driver implementation.
Definition stm32_adc.hpp:23
STM32ADC(ADC_HandleTypeDef *hadc, RawData dma_buff, std::initializer_list< uint32_t > channels, float vref)
构造 ADC 驱动对象 / Construct ADC driver object
Definition stm32_adc.cpp:48
~STM32ADC()
析构函数 / Destructor
Definition stm32_adc.cpp:95
float ReadChannel(uint8_t channel)
读取指定通道电压 / Read channel voltage
Channel & GetChannel(uint8_t index)
获取 ADC 通道对象 / Get ADC channel object
LibXR 命名空间
Definition ch32_can.hpp:14
void STM32_InvalidateDCacheByAddr(const void *addr, size_t size)
Invalidates D-Cache lines covering the specified memory range.