MSP430F5438 ADC12学习笔记

发布者:码梦创想最新更新时间:2017-02-20 来源: eefocus关键字:MSP430F5438  ADC12 手机看文章 扫描二维码
随时随地手机看文章

1.前言

这几天实践了MSP430的ADC12功能,虽然片内AD功能比较简单但是还学出了点“门道”来,这个“门道”便是MSP430F5438A和MSP430F5438的区别。这里通过一个例子说明片内ADC的使用,首先实现UART和定时器1S溢出的功能,在上述功能的基础上每1S打印一次AD转换结果,转换通道定向到通道11,该通道对应AVCC和AVSS插值的一半,由于AVCC和LDO的输出之间只有一个电感连接,可以理解转换的结果为LDO输出电压的一般,若扩大两倍便是LDO的实际输出结果,在本文所用的开发板LDO输出为3.3V,所有打印的结果越接近3.3V越好。

2.代码实现和输出结果

代码实现


  1. // 时钟默认情况  

  2. // FLL时钟      FLL选择 XT1  

  3. // 辅助时钟     ACLK选择 XT1          32768Hz  

  4. // 主系统时钟   MCLK选择 DCOCLKDIV    8000000Hz  

  5. // 子系统时钟   SMCLK选择 DCOCLKDIV   8000000Hz  

  6. // TA1选择ACLK,最大计数值为32768,中断频率为1HZ  

  7.   

  8. #include   

  9. #include   

  10. #include   

  11. void clock_config(void);  

  12. void select_xt1(void);  

  13. void dco_config(void);  

  14. void adc12_config(void);  

  15. void uart_config(void);  

  16. char second_flag  = 0;                          // 1S标志  

  17.   

  18. int main(void)  

  19. {  

  20.     clock_config();                             // 初始化时钟  

  21.     adc12_config();                             // 初始化ADC12  

  22.     uart_config();  

  23.   

  24.     TA1CCTL0 = CCIE;                            // 使能TA1CCR0,比较匹配中断  

  25.     TA1CCR0 = 32768;                            // 初始化最大值,发生比较匹配中断频率 32768/32768 = 1Hz  

  26.     TA1CTL = TASSEL_1 + MC_1 + TACLR;           // 选择ACLK,最大值为CCR0,清除计数值  

  27.   

  28.     _EINT();                                    // 初始化全局中断  

  29.   

  30.     while(1)  

  31.     {  

  32.         if( second_flag )  

  33.         {  

  34.             second_flag = 0;                        // 1s时间到  

  35.   

  36.             ADC12CTL0 |= ADC12SC;                   // 启动转换  

  37.             while ( !(ADC12IFG & BIT0) );           // 等待转换完成  

  38.   

  39.             // 被转换的通道为通道11 (AVCC-AVSS)/2;  

  40.             // 此时转换的精度为12位——4096  

  41.             // AVCC通过一个电感和LDO的输出端连接  

  42.             // 打印LDO输出电压,保留3位精度  

  43.             float ldo_voltage = ADC12MEM0  / 4096.0 * 3.3 * 2;  

  44.             printf("LDO Voltage %.3f\r\n",ldo_voltage);  

  45.         }  

  46.     }  

  47. }  

  48.   

  49. void clock_config(void)  

  50. {  

  51.     WDTCTL = WDTPW + WDTHOLD;                   // 停止看门狗  

  52.     select_xt1();                               // 选择XT1  

  53.     dco_config();                               // ACLK = XT1 = 32.768K  

  54.                                                 // MCLK = SMCLK = 8000K  

  55. }  

  56.   

  57. void select_xt1(void)  

  58. {  

  59.     // 启动XT1  

  60.     P7SEL |= 0x03;                              // P7.0 P7.1 外设功能  

  61.     UCSCTL6 &= ~(XT1OFF);                       // XT1打开  

  62.     UCSCTL6 |= XCAP_3;                          // 内部电容  

  63.     do  

  64.     {  

  65.         UCSCTL7 &= ~XT1LFOFFG;                  // 清楚XT1错误标记  

  66.     }while (UCSCTL7&XT1LFOFFG);                 // 检测XT1错误标记  

  67. }  

  68.   

  69. void dco_config(void)  

  70. {  

  71.     __bis_SR_register(SCG0);                    // 禁止FLL功能  

  72.     UCSCTL0 = 0x0000;                           // Set lowest possible DCOx, MODx  

  73.     UCSCTL1 = DCORSEL_5;                        // DCO最大频率为16MHz  

  74.     UCSCTL2 = FLLD_1 + 243;                     // 设置DCO频率为8MHz  

  75.                                                 // MCLK = SMCLK= Fdcoclkdiv = (N+1)X(Ffllrefclk/n)  

  76.                                                 // N为唯一需要计算的值  

  77.                                                 // Ffllrefclk FLL参考时钟,默认为XT1  

  78.                                                 // n取默认值,此时为1  

  79.                                                 // (243 + 1) * 32768 = 8MHz  

  80.     __bic_SR_register(SCG0);                    // 使能FLL功能  

  81.   

  82.     // 必要延时  

  83.     __delay_cycles(250000);  

  84.   

  85.     // 清楚错误标志位  

  86.     do  

  87.     {  

  88.         UCSCTL7 &= ~(XT2OFFG + XT1LFOFFG + XT1HFOFFG + DCOFFG);  

  89.                                                 // 清除所有振荡器错误标志位  

  90.         SFRIFG1 &= ~OFIFG;                      // 清除振荡器错误  

  91.     }while (SFRIFG1&OFIFG);                     // 等待清楚完成  

  92. }  

  93.   

  94. void adc12_config(void)  

  95. {  

  96.     // 只有在ADC12ENC复位的情况下才可以操作  

  97.     // ADC12SHT1X ADC12SHT0X ADC12MSC ADC12REF2_5V ADC12REFON ADC12ON  

  98.     ADC12CTL0 &= ~ADC12ENC;  

  99.   

  100.     // 设置采样保持时间,最大时间周期以提高转换精度  

  101.     // 注意MSP430F5438没有REF模块,片内基准无效  

  102.     // 操作ADC12REF2_5V ,ADC12REFON并无意义  

  103.     ADC12CTL0 = ADC12SHT0_15 + ADC12SHT1_15 + ADC12ON;  

  104. //    ADC12CTL0 = ADC12SHT0_15 + ADC12SHT1_15 + ADC12ON +  

  105. //                ADC12REF2_5V + ADC12REFON;  

  106.     // 采样保持脉冲来自采样定时器  

  107.     ADC12CTL1 = ADC12SHP;  

  108.     // 关闭内部内部温度检测以降低功耗,注意或操作否则修改转换精度  

  109.     ADC12CTL2 |= ADC12TCOFF ;  

  110.     // 基准电压选择AVCC,并选择11通道——(AVCC-AVSS)/2  

  111.     ADC12MCTL0 = ADC12SREF_0 + ADC12INCH_11;  

  112.   

  113.     __delay_cycles(75);  

  114.     // ADC12使能  

  115.     ADC12CTL0 |= ADC12ENC;  

  116. }  

  117.   

  118. void uart_config(void)  

  119. {  

  120.     P3SEL = 0x30;                               // 选择P3.4和P3.5的复用功能  

  121.   

  122.     UCA0CTL1 |= UCSWRST;                        // 软件复位  

  123.     UCA0CTL1 |= UCSSEL_1;                       // 选择ACLK时钟  

  124.     UCA0BR0 = 3;                                // 查表获得  

  125.     UCA0BR1 = 0;                                // UCA0BRX和UCA0MCTL数值  

  126.     UCA0MCTL |= UCBRS_3 + UCBRF_0;              //  

  127.     UCA0CTL1 &= ~UCSWRST;                       //  

  128.   

  129.     UCA0IE |= UCRXIE;                           // 使能接收中断  

  130. }  

  131.   

  132. int putchar(int ch)  

  133. {  

  134.     UCA0TXBUF = ch;  

  135.     while(!(UCA0IFG & UCTXIFG));  

  136.     return ch;  

  137. }  

  138.   

  139. #pragma vector=TIMER1_A0_VECTOR  

  140. __interrupt void TIMER1_A0_ISR(void)  

  141. {  

  142.     second_flag = 1;  

  143. }  


图1 参考电压AVCC(3.3V)

3.一些注意点

3.1 提高采样时间

如果条件允许,可以尽可能的提高采样时间,这样转换结果可以更稳定一些。

3.2 MSP430F5438没有REF模块

现在(2013年10月)可以在TI官网上下载得到的示例代码或数据手册参考手册等,都是围绕MSP430F5438A的。但是市面上很多MCU还是MSP430F5438,其实MSP430F5438A和MSP4305438是有区别的,MSP430F5438没有REF模块,所以使用片内的2.5参考电源还是有些不稳定的因素。可以通过以下的实现测试一下,AD转换的目标依然是LDO输出。
需要修改以下几个部分的代码
第一:
ADC12CTL0 = ADC12SHT0_15 + ADC12SHT1_15 + ADC12ON; 修改为
ADC12CTL0 = ADC12SHT0_15 + ADC12SHT1_15 + ADC12ON +
                       ADC12REF2_5V + ADC12REFON;
使用打开片内2.5V参考电源
第二:
ADC12MCTL0 = ADC12SREF_0 + ADC12INCH_11;修改为
ADC12MCTL0 = ADC12SREF_1 + ADC12INCH_11;
转换参考电压为Vref,即修改1设置的2.5V参考电源

第三:
float ldo_voltage = ADC12MEM0 / 4096.0 * 3.3 * 2;修改为
float ldo_voltage = ADC12MEM0 / 4096.0 * 2.5 * 2;
替换转换公式,参考电压由3.3V变为2.5V

输出结果如下,结果发现LDO的输出电压为3.4V,比实际电压高0.1V。

图2 参考电压VREF(2.5V)

图3 MSP430参考手册说明


关键字:MSP430F5438  ADC12 引用地址:MSP430F5438 ADC12学习笔记

上一篇:MSP430F5438中断函数两种编写方法
下一篇:如何在FreeRTOS下实现低功耗——MSP430F5438平台

推荐阅读最新更新时间:2024-03-16 15:34

MSP430F5529 DriverLib 库函数学习笔记(八)模数转换模块(ADC12)
平台:Code Composer Studio 10.3.1 MSP430F5529 LaunchPad™ Development Kit (MSP‑EXP430F5529LP) 硬知识 模数转换概述 在MSP430单片机的实时控制和智能仪表等实际应用系统中,常常会遇到连续变化的物理量,如温度、流量、压力和速度等。利用传感器把这些物理量检测出来,转换为模拟电信号,再经过模数转换模块(ADC)转换成数字量,模拟量才能够被MSP430单片机处理和控制。 1.模数转换基本过程 首先连续时间输入信号x(t)输入ADC的采样保持器中,ADC每隔Ts(采样周期)读出一次x(t)的采样值,对此采样值进行量化。量化的过程是将此
[单片机]
MSP430F5529 DriverLib 库函数学习笔记(八)模数转换模块(<font color='red'>ADC12</font>)
msp430的一些常用的C语言控制程序---ADC12(3)
下面是ADC12的序列通道单次转化的例子: //ADC12序列通道单次 #include msp430x14x.h unsigned ADC_Result1,ADC_Result2; void ADC12_Init(); void main(){ WDTCTL = WDTPW + WDTHOLD; ADC12_Init(); ADC12CTL0 |= ADC12SC; _BIS_SR(GIE); while(1); } // void ADC12_Init(){ P6DIR &= (~BIT6 + ~BIT7);//设置状态为输入态 P6SEL |= BIT6 + BIT7;//选择端口的第二功能 AD
[单片机]
MSP430F5438学习笔记 TA1溢出中断加比较匹配中断
// 时钟默认情况 // FLL时钟 FLL选择 XT1 // 辅助时钟 ACLK选择 XT1 32768Hz // 主系统时钟 MCLK选择 DCOCLKDIV 8000000Hz // 子系统时钟 SMCLK选择 DCOCLKDIV 8000000Hz // TA1选择ACLK,最大计数值为65535 // 比较匹配值为 32768 // 在TIMER1_A0_VECTOR中 CCR0中断 P4.0 = 0 // 在TIMER1_A1_VECTOR中 OVF中断 P4.0 = 1 // 实际效果 P4.0 1s为高电平,1s为低电平,交替进行 #include
[单片机]
MSP430F5438学习笔记 UART ACLK 9600-8-N-1
1.初始化UART0之前需要先初始化ACLK、SMCLK和MCLK。示例代码中使用XT1,ACLK为32768,SMCLK和MCLK约为8MHZ。 2.UART的时钟可以参考ACLK或者SMCLK,本例参考ACLK。由于参考ACLK时钟,所以串口速率不能超过32768。选择9600较为合适。 3.MSP430波特率的产生有两种模式,低频波特率产生和过采样波特率产生。代码中使用低频波特率产生。 4.代码的开头调用了stdio,在函数中宏重写了putchar函数,定向到UART单字节输出。 5.代码初始化之后输出 Hello MSP430,随后直接反射串口接收到的数据,例如发送123456即返回123456。 /
[单片机]
msp430F5438+CC2520通信
软件的实现主要是对硬件的初始化和简单基于IEEE820.15.4格式的无线收发。 硬件初始化包括时钟的初始化,SPI初始化,UART初始化等。 以下是一个时钟初始化的程序: MSP430F5438 // ----------------- // /|\| | // | | P7.0|-- XT1 37.768K // --| P7.1 |-- // | P5.2|-- XT2 16M // | P5.3|-- // | | 硬件连接方面,P7.0与P7.1连接晶振XTI,P
[单片机]
MSP430F5438研究心得
一、多源中断问题 #pragma vector = PORT2_VECTOR __interrupt void port2(void) { switch(P2IV) { case P2IV_P2IFG6: P2IFG &=~BIT6; P1OUT ^= BIT0;break; //LED1 亮灭 case P2IV_P2IFG7: P2IFG &=~BIT7; P1OUT ^= BIT1;break; //LED2 亮灭 default :break; } } #pragma vector = PORT2_VECTOR __inter
[单片机]
MSP430F5438研究心得
一、多源中断问题 #pragma vector = PORT2_VECTOR __interrupt void port2(void) { switch(P2IV) { case P2IV_P2IFG6: P2IFG &=~BIT6; P1OUT ^= BIT0;break; //LED1 亮灭 case P2IV_P2IFG7: P2IFG &=~BIT7; P1OUT ^= BIT1;break; //LED2 亮灭 default :break; } } #pragma vector = PORT2_VECTOR __inter
[单片机]
MSP430F5438 看门狗实验
简介: 1,学会在定时器和看门狗模式下操作 WDT,了解 WDT 寄存器的配置 2,编程实现WDT 定时功能,WDT 时钟源来自 ACLK,且 ACLK 无失效来自 XT1 晶体 3,实现功能:250ms 定时,每 250ms LED1 亮灭交替闪烁 1,看门狗介绍 看门狗定时器是一个32位的定时器它可以作为看门狗或定时器使用所有F5XX系列的器件中都有增强型看门狗定时器—WDT_A。 什么是看门狗? 从字面的意思理解,看门狗帮我的系统看门,防止我的系统到处乱跑。在实际嵌入式系统中看门狗也是充当这个角色的。看门狗,简称WDT,很明显它实际上是一个定时器,就如同我们所讲的Timer_A3亦或是Timer_B7系统一样,
[单片机]
<font color='red'>MSP430F5438</font> 看门狗实验
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved