ADC类 – 模数转换¶
使用:
import pyb
adc = pyb.ADC(pin) # create an analog object from a pin 从引脚中创建一个模拟对象
val = adc.read() # read an analog value 读取一个模拟对象
adc = pyb.ADCAll(resolution) # create an ADCAll object 创建一个ADCALL对象
val = adc.read_channel(channel) # read the given channel 读取给定通道
val = adc.read_core_temp() # read MCU temperature 读取MCU温度
val = adc.read_core_vbat() # read MCU VBAT 读取MCU VBAT
val = adc.read_core_vref() # read MCU VREF 读取MCU CREF
构造函数¶
-
class
pyb.
ADC
(pin) 创建一个与给定引脚相关联的ADC对象,然后您就可在引脚中读取模拟值。
方法¶
-
ADC.
read
() 在模拟引脚上读取值并返回。返回值应介于0至4095之间。
-
ADC.
read_timed
(buf, timer) 以
timer
对象设定的比率将模拟值读取到buf
中。buf
可为字节数组或数组。ADC值有12位分辨率,其中元素大小大于等于16位时,直接储存到buf
中。 若buf
仅有8位元素(例如一个字节元组),则样本分辨率将会降低到8位。timer
应为一个定时器对象,定时器每次触发时,样本都会被读取。定时器须已初始化,且以预设的样本频率运行。为支持此函数之前的活动,
timer
可为一个指定样本频率(以Hz为单位)的整数。在此种情况下,定时器(6)可自动配置为以给定频率运行。定时器对象用法示例(最佳方法):
adc = pyb.ADC(pyb.Pin.board.X19) # create an ADC on pin X19 在X19引脚上创建一个ADC tim = pyb.Timer(6, freq=10) # create a timer running at 10Hz 创建一个以10Hz运行的定时器 buf = bytearray(100) # creat a buffer to store the samples 创建一个储存样本的缓冲区 adc.read_timed(buf, tim) # sample 100 values, taking 10s 样本100个值,耗时10s
Example using an integer for the frequency:
adc = pyb.ADC(pyb.Pin.board.X19) # create an ADC on pin X19 在引脚X19上创建一个ADC buf = bytearray(100) # create a buffer of 100 bytes 创建100字节的缓冲区 adc.read_timed(buf, 10) # read analog values into buf at 10Hz 以10Hz将模拟值读取到缓冲区中 # this will take 10 seconds to finish 耗时10s后结束 for val in buf: # loop over all values 循环所有值 print(val) # print the value out 打印值
此函数不分配任何内存。
ADCAll对象¶
这一实例将所有ADC引脚转换为模拟输入。原始的MCU温度、VREF和VBAT数据可分别在ADC通道16、17和18中访问。需要适度的缩放。芯片上的温度传感器绝对精度不高,只适用于检测温度变化。
ADCAll
read_core_vbat()
和 read_core_vref()
使用3.3 V电源作为参考,读取备用电池电压和(1.21 V)参考电压。
假定 ADCAll
对象已通过 adc = pyb.ADCAll(12)
实例化,则3.3伏电源电压可计算为:
v33 = 3.3 * 1.21 / adc.read_core_vref()
若3.3V电压正确,则 adc.read_core_vbat()
值有效。若电源电压降至3.3V以下,例如在在电池供电的放电电池系统中,该稳压器将无法保持3.3V电源,导致错误读取。此时若要产生一个有效值,请遵循以下步骤:
vback = adc.read_core_vbat() * 1.21 / adc.read_core_vref()
在不产生 ADCAll
副作用的前提下访问该值并非不可能:
def adcread(chan): # 16 temp 17 vbat 18 vref 温度16电池电压17参考电压18
assert chan >= 16 and chan <= 18, 'Invalid ADC channel'
start = pyb.millis()
timeout = 100
stm.mem32[stm.RCC + stm.RCC_APB2ENR] |= 0x100 # enable ADC1 clock.0x4100
stm.mem32[stm.ADC1 + stm.ADC_CR2] = 1 # Turn on ADC 启动ADC
stm.mem32[stm.ADC1 + stm.ADC_CR1] = 0 # 12 bit
if chan == 17:
stm.mem32[stm.ADC1 + stm.ADC_SMPR1] = 0x200000 # 15 cycles 15周期
stm.mem32[stm.ADC + 4] = 1 << 23
elif chan == 18:
stm.mem32[stm.ADC1 + stm.ADC_SMPR1] = 0x1000000
stm.mem32[stm.ADC + 4] = 0xc00000
else:
stm.mem32[stm.ADC1 + stm.ADC_SMPR1] = 0x40000
stm.mem32[stm.ADC + 4] = 1 << 23
stm.mem32[stm.ADC1 + stm.ADC_SQR3] = chan
stm.mem32[stm.ADC1 + stm.ADC_CR2] = 1 | (1 << 30) | (1 << 10) # start conversion开始转换
while not stm.mem32[stm.ADC1 + stm.ADC_SR] & 2: # wait for EOC
if pyb.elapsed_millis(start) > timeout:
raise OSError('ADC timout')
data = stm.mem32[stm.ADC1 + stm.ADC_DR] # clear down EOC 清空EOC
stm.mem32[stm.ADC1 + stm.ADC_CR2] = 0 # Turn off ADC 关闭ADC
return data
def v33():
return 4096 * 1.21 / adcread(17)
def vbat():
return 1.21 * 2 * adcread(18) / adcread(17) # 2:1 divider on Vbat channel Vbat通道的2:1分频器
def vref():
return 3.3 * adcread(17) / 4096
def temperature():
return 25 + 400 * (3.3 * adcread(16) / 4096 - 0.76)