设为首页 收藏本站
查看: 1544|回复: 0

[经验分享] MSP430 入门学习资料

[复制链接]

尚未签到

发表于 2015-11-5 13:30:50 | 显示全部楼层 |阅读模式
一,MSP430G2553单片机的各个功能模块
  (一),IO口模块,
  1,我们所用的MSP430G2553有两组IO口,P1P2
  2IO口的寄存器有:方向选择寄存器PxDIR,输出寄存器PxOUT,输入寄存器PxINIO口内部上拉或下拉电阻使能寄存器PxRENIO口功能选择寄存器PxSELPxSEL2IO口中断使能寄存器PxIE,中断沿选择寄存器PxIESIO口中断标志寄存器PxIFG
  3,所有的IO都带有中断,其中所有的P1口公用一个中断向量,所有的P2口公用一个中断向量。所以在使用中断时,当进入中断后,还要判断到底是哪一个IO口产生的中断,判断方法可以是判断各个IO口的电平。
   4,中断标志PxIFG需要软件清除,也可以用软件置位,从而用软件触发一个中断。
注意:在设置PxIESx时根据PxINx有可能会引起相应的PxIFGx置位(具体的情况见用户指南),所以在初始化完IO口中断以后,正式使用IO中断前要先将对应的PxIFGx清零。程序如下:
void IO_interrupt_init()     //IO中断初始化函数
{
  P1REN |= BIT4+BIT5+BIT6+BIT7;     // pullup 内部上拉电阻使能
  //使用中断时,使能内部的上拉电阻这样当该脚悬空是,电平不会跳变,防止悬空时电平跳变不停的触发中断
  P1OUT = BIT4+BIT5+BIT6+BIT7;   // 当引脚上的上拉或下拉电阻使能时,PxOUT选择是上拉还是下来
          //0:下拉,1:上拉

  P1IE |= BIT4+BIT5+BIT6+BIT7;       // interrupt enabled P13中断使能
  P1IES |= BIT4+BIT5+BIT6+BIT7;          // Hi/lo edge  下降沿中断
  //P1IES &= ~BIT3;                         //上升沿触发中断
  P1IFG &= ~(BIT4+BIT5+BIT6+BIT7);         //中断标志位清零

}

5PxOUT:如果引脚选择了内部的上拉或下拉电阻使能,则PxOUT设定电阻是上拉还是下拉,0:下拉,1:上拉
6,当IO口不用时,最好不要设为输入,且为浮动状态(这是IO口的默认状态),因为当输入为浮动时,输入电压有可能会在VILVIH之间,这样会产生击穿电流。所以不用的IO口可以设为输出状态,或设为输入状态但通过外围电路接至VCCGND,或接一个上拉/下拉电阻。
7,当使用msp430g2553IO口时要注意,因为g2553IO口寄存器的操作,不像51,它不能单独针对某一位进行操作,必须对整个寄存器进行操作。所以就不像51g2553不可以定义bit型的数据。所以在使用mspIO口时要注意对需要位的操作,而不要影响其他无关的位,可以 用 |  &  ^等按位操作的符号。在使用IO都控制其他外围模块时也要注意要使用的IO口的定义,可以用如下的定义方法:
#define CLR_RS P2OUT&=~BIT0;    //RS = P2.0
#define SET_RS P2OUT|=BIT0;
#define CLR_RW P2OUT&=~BIT1; //RW = P2.1
#define SET_RW P2OUT|=BIT1;
#define CLR_EN P2OUT&=~BIT2; //EN = P2.2
#define SET_EN P2OUT|=BIT2;

#define DataPort    P1OUT

8g2553P27P26脚分别接外部晶体的输出和输入脚XOUTXIN,默认是自动设为了晶振管脚功能,但是当想把它们用为普通的IO时,也可以,设置对应的SEL设为普通的IO即可,如下:
P2DIR |= BIT6+BIT7;    //P26P27配置为普通IO 并为输出脚  默认为晶振的输入和输出引脚 作为dac0832
    P2SEL &= ~(BIT6+BIT7);   //cswr控制端
    P2SEL2 &= ~(BIT6+BIT7);



(二),时钟系统
     1msp430能做到超低功耗,合理的时钟模块是功不可没的。但是功能强大的时钟模块设置起来也相对复杂一些。
     2msp430的时钟源有:
(1),外接低频晶振LFXT1CLK:低频模式接手表晶体32768Hz,高频模式450KHz~8MHz
2),外接高速晶振XT2CLK8MHz
3),内部数字控制振荡器DCO:是一个可控的RC振荡器,频率在0~16MHz
4),超低功耗低频振荡器VLO:不可控,4~20KHz 典型值为12KHz
     3,时钟模块:430的时钟模块有MCLK  SMCLK  ACLK
(1),主系统时钟MCLK:提供给MSP430CPU时钟。可以来自LFXT1CLK  XT2CLK  DCO  VLO可选,默认为DCO
(2),子系统时钟SMCLK: 提供给高速外设。可以来自LFXT1CLK  XT2CLK  DCO  VLO可选,默认为DCO
(3),辅助系统时钟ACLK:提供给低速外设。可来自LFXT1CLK  VLO
     4,内部的振荡器DCOVLO提供的时钟频率不是很精确,随外部环境变化较大。
DCO默认的频率大概为800KHz,但我用示波器观察的为1.086MHz左右,当DCO设置的过高时,用示波器可以看到波形不再是方波,而是类似于正弦波。DCO可以用CCS提供的宏定义进行相对比较精确的设置,如下:
DCOCTL = CALDCO_12MHZ;   //DCO设为12MHz   这种方法设DCO频率比较精确,实际测得为12.08MHz左右 正弦波
BCSCTL1 = CALBC1_12MHZ;
用这种方法可以设置1,8,12,16MHz
宏定义如下:
#ifndef __DisableCalData

SFR_8BIT(CALDCO_16MHZ);                       
SFR_8BIT(CALBC1_16MHZ);                       
SFR_8BIT(CALDCO_12MHZ);                       
SFR_8BIT(CALBC1_12MHZ);                       
SFR_8BIT(CALDCO_8MHZ);                        
SFR_8BIT(CALBC1_8MHZ);                        
SFR_8BIT(CALDCO_1MHZ);                        
SFR_8BIT(CALBC1_1MHZ);                        

#endif

5,使用超低功耗低频振荡器VLO可以很大程度地降低系统功耗,下面的例子是设置ACLKVLOMCLKVLO8分频:
#include  <msp430g2553.h>

//1延时
//#define CPU_F ((double)16000000)//cpu frequency16000000
#define CPU_F ((double)1630)//cpu frequency1630   //CPU的实际MCLK大约为13.05/8=1.63KHz
#define delay_us(x) __delay_cycles((long)(CPU_F*(double)x/1000000.0))
#define delay_ms(x) __delay_cycles((long)(CPU_F*(double)x/1000.0))


void main(void)
{
  volatile unsigned int i;                  // Volatile to prevent removal
  WDTCTL = WDTPW &#43; WDTHOLD;                 // Stop watchdog timer

  BCSCTL3 |= LFXT1S_2;                      // LFXT1 = VLO  低频时钟选择为VLO ACLK选为VLO

  IFG1 &= ~OFIFG;                           // Clear OSCFault flag  清除振荡器错误中断标志

  __bis_SR_register(SCG1 &#43; SCG0);           // Stop DCO  SCG1禁止SMCLK  SCG0禁止DCO

  BCSCTL2 |= SELM_3 &#43; DIVM_3;               // MCLK = LFXT1/8
  //因为前面已经选择了LFXT1 = VLO 所以MCLK选为VLO  8分频  所以CPU的MCLK大约为1.5KHz

  P1DIR = 0xFF;                             // All P1.x outputs
  P1OUT = 0;                                // All P1.x reset
  P2DIR = 0xFF;                             // All P2.x outputs
  P2OUT = 0;                                // All P2.x reset

  P1SEL |= BIT0&#43;BIT4;                // P10 P14options  功能选择为外围模块
  //p10输出ACLK,来自VLO,p14输出SMCLK,  因为禁止了SMCLK,所以P14脚无波形输出
  //VLO典型&#20540;为12KHz  实际用示波器测得为:13.05KHz 左右波动
  //所以CPU的实际MCLK大约为13.05/8=1.63KHz


  for (;;)
  {
    P1OUT ^= BIT6;          // P1.6 闪烁
    delay_ms(1000);
  }
}

  6,如上面的程序所示,其中的延迟函数用那种方法,使用系统的延迟周期函数__delay_cycles(int n); 可以达到比较精确的延迟,如下:

//more_
//1延时
//#define CPU_F ((double)16000000)//cpu frequency16000000
#define CPU_F ((double)12000000)//cpu frequency12000000
#define delay_us(x) __delay_cycles((long)(CPU_F*(double)x/1000000.0))
#define delay_ms(x) __delay_cycles((long)(CPU_F*(double)x/1000.0))
//2空函数
#define nop() _NOP();

   7,系统上电后默认使用的是DCO时钟,DCO默认的频率大概为800KHz,但我用示波器观察的为1.086MHz左右,当DCO设置的过高时,用示波器可以看到波形不再是方波,而是类&#20284;于正弦波。



(三),定时器Timer_A
   1MSP430g2553具有两个16位的定时器:Timer0_A   Timer1_A。分别具有三个捕捉/比较寄存器,具有输入捕捉,输出比较功能。可以产生定时中断,也可以产生PWM
   2,产生PWM,例子如下:

#include <msp430g2553.h>

void Timer_A0_1_init()  //TA0.1输出PWM
{
TACTL|= TASSEL_1&#43;MC_1;//ACLK,增计数
CCTL1=OUTMOD_7;//输出模式为复位/置位
CCR0=328;//时钟频率为32768HZ,100HZ
//CCR1=164;//时钟频率为32768HZ,占空比CCR1/CCR0=50%
CCR1=109;//占空比CCR1/CCR0=1/3                              TA0.1由P1.2 P1.6输出
}

void Timer_A1_2_init()    //TA1.2输出PWM
{
TA1CTL|= TASSEL_1&#43;MC_1;//ACLK,增计数
TA1CCTL2=OUTMOD_7;//输出模式为复位/置位,注意CCTL2要写为TA1CCTL2

TA1CCR0=164;//时钟频率为32768HZ,波形32768/CCR0=199HZ
TA1CCR2=41;//占空比CCR2/CCR0=1/4,注意CCR2要写成TA1CCR2       TA1.2由P2.4 P2.5输出

}

void Timer_A1_1_init()   //TA1.1输出PWM
{
TA1CCTL1=OUTMOD_7;
TA1CCR1=123;   //占空比CCR1/CCR0=3/4,注意CCR1要写成TA1CCR1   TA1.1由P2.1 P2.2输出
}

void IO_init()
{
P1SEL|=BIT2&#43;BIT6;
P1DIR|=BIT2&#43;BIT6;//P1.2 P1.6输出   TA0.1   OUT1

P2SEL|=BIT4&#43;BIT5;
P2DIR|=BIT4&#43;BIT5;//P2.4 P2.5输出   TA1.2   OUT2

P2SEL|=BIT1&#43;BIT2;
P2DIR|=BIT1&#43;BIT2; //P2.1 P2.2输出  TA1.1   OUT1
}

void main(void) {
WDTCTL=WDTPW&#43;WDTHOLD;

    IO_init();

Timer_A0_1_init();
Timer_A1_2_init();
Timer_A1_1_init();


_BIS_SR(CPUOFF);                          // Enter LPM0  进入低功耗模式0 SMCLK ON,ACLK ON
}

  3Timer_A的捕获/比较寄存器
TAR寄存器是Timer_A16位的计数寄存器。TACCRxTimer_A的捕获/比较寄存器,当为捕获模式时:当捕获发生时,把TAR的&#20540;装载到TACCRx中。当为比较模式时:TACCRx中装的是要与TAR寄存器相比较的&#20540;。
   4,捕获模式
捕获外部输入的信号的上升沿或下降沿或上升沿下降沿都捕捉,当捕捉发生时,把TAR的&#20540;装载到TACCRx中,同时也可以进入中断,执行相应的操作。这样利用捕捉上升沿或下降沿就可以计算外部输入信号的周期,得出频率。利用捕捉上升沿和下降沿可以得出输入信号的高电平或低电平的持续时间。也可以算出占空比。下面是一个例子,是Timer_A捕获初始化的程序:
void timer_init()      //使用Timer1_A时要特别注意各个寄存器的写法,因为Timer0_A的寄存器都简写了,所以在写
//Timer1_A的寄存器时,要特别注意与Timer0_A的不同
{
P1SEL |= BIT2;    //选择P12作为捕捉的输入端子  Timer0_A
//TACCTL1 |=CM_3&#43;SCS&#43;CAP&#43;CCIE;  //上下沿都触发捕捉,用于测脉宽,同步模式、时能中断  CCI1A
TACCTL1 |=CM_1&#43;SCS&#43;CAP&#43;CCIE;  //上升沿触发捕捉,同步模式、时能中断  CCI1A
    TACTL |= TASSEL1&#43;MC_2;  //选择SMCLK时钟作为计数时钟源,不分频   增计数模式不行,必须连续计数模式

  P2SEL |= BIT1;    //选择P21作为捕捉的输入端子    Timer1_A
  //TA1CCTL1 |=CM_3&#43;SCS&#43;CAP&#43;CCIE;  //上下沿都触发捕捉,用于测脉宽,同步模式、时能中断  CCI1A
  TA1CCTL1 |=CM_1&#43;SCS&#43;CAP&#43;CCIE;  //上升沿触发捕捉,同步模式、时能中断  CCI1A
  TA1CTL |= TASSEL1&#43;MC_2;  //选择SMCLK时钟作为计数时钟源,不分频   增计数模式不行,必须连续计数模式

}


相对应的中断函数如下:

#pragma vector=TIMER0_A1_VECTOR   //Timer0_A CC1  的中断向量
__interrupt void Timer_A(void)
{

// CCI0A 使用的捕捉比较寄存器是TA0CCR0TA0CCR0单独分配给一个
    //中断向量TIMER1_A0_VECTOR,所以进入中断后直接就是Timer0_A CC0产生的中断,不用经过类&#20284;
   //下面的方法判断中断源了  。
//Timer0_A CC1-4, TA0公用一个中断向量 TIMER0_A1_VECTOR,所以进入了中断后还要用下面
    //的方法进行判断是哪一个中断源产生的中断
  switch(TAIV)    //如果是Timer0_A CC1产生的中断
  {
  case 2:
   {
   flag=1;
   LPM1_EXIT;      //退出低功耗模式
  // _BIC_SR_IRQ(LPM1_bits);
   //_bic_SR_register_on_exit(LPM1_bits);
   break;
   }
  case 4: break;
  case 10:break;
  }
}


#pragma vector=TIMER1_A1_VECTOR      //Timer1_A CC1  的中断向量
__interrupt void Timer_A1(void)
{

// P1OUT|=BIT0;  //led调试用的
// LPM1_EXIT;      //退出低功耗模式  因为使用的是CCI0A 使用的捕捉比较寄存器是TA1CCR0TA1CCR0单独分配给一个
                //中断向量TIMER1_A0_VECTOR,所以进入中断后直接就是Timer1_A CC0产生的中断,不用经过类&#20284;
               //下面注释掉的方法判断  。
               //Timer1_A CC1-4, TA1则公用一个中断向量 TIMER1_A1_VECTOR,所以进入了中断后还要用下面
       //的方法进行判断是哪一个中断源产生的中断
  switch(TA1IV)    //如果是Timer1_A CC1产生的中断
  {
  case 2:
   {
   flag=2;
   LPM1_EXIT;      //退出低功耗模式

  // _BIC_SR_IRQ(LPM1_bits);
   //_bic_SR_register_on_exit(LPM1_bits);
   break;
   }
  case 4:break;
  case 10:break;
  }
}

//如果要测量更低频率的信号的话,可以在中断中判断溢出中断发生的次数,这样就可以得到溢出的次数,从而可以测量更
//低频率的信号

  5Timer_A的计数模式
计数模式有:增计数模式,连续计数模式和增减计数模式。具体的各个模式的详解,参见用户指南。
   6,定时器的定时中断
在使用定时器的定时中断时,要注意定时器计数模式的选择。在使用中断时,要注意中断向量的使用和中断源的判断,下面就举一个例子,注释的也较详细:
#include <msp430g2553.h>

unsigned int t=0;

void main(void)
{
  WDTCTL = WDTPW &#43; WDTHOLD;                 // Stop WDT
  P1DIR |= 0x01;                            // P1.0 output

  CCTL0 = CCIE;   // CCTLx是捕获/比较控制寄存器   interrupt enabled  CCIE=0x0010  时能定时器A中断

  CCR0 = 50000;   //捕获/比较寄存器   设置计数器CCR0的初&#20540;  16位寄存器,最大&#20540;为65535
                //默认SMCLK使用的是DCO,默认的DCO大约为800KHz,而CCR0=50000,所以中断产生的频率大约为16Hz
  TACTL = TASSEL_2 &#43; MC_2;                  // SMCLK, contmode  连续计数模式从0计到0FFFFh
  //TACTL = TASSEL_2 &#43; MC_1;                  // SMCLK, upmode  增计数模式从0计到CCR0

  _BIS_SR(LPM0_bits &#43; GIE);                 // Enter LPM0 w/ interrupt  进入低功耗模式0,允许中断
}

// Timer A0 interrupt service routine
#pragma vector=TIMER0_A0_VECTOR
__interrupt void Timer_A (void)    //CCIFG中断被响应后,该标志位自动清零

{
  //P1OUT ^= 0x01;                            // Toggle P1.0
t&#43;&#43;;
if(t==5)
{
P1OUT ^= BIT0;           // Toggle P1.0
t=0;
}

  CCR0 &#43;= 50000;                            // Add Offset to CCR0  增加CCR0偏移

   
  //定时器总是从0开始往上计数,一直到计满再从0开始,在连续计数模式下,当定时器的&#20540;等于CCR0时,产生中断
  //在中断中对CCR0增加50000,这样的话定时器从当前&#20540;到下一时刻再次等于CCR0时的间隔为50000,恒定
  //这样产生中断的时间间隔就相等了
//所以在连续计数模式下,要想使中断的时间间隔一定,就要有CCR0 &#43;= n;这句话
                                              //在中断中CCR0不需要从新赋&#20540;,区别于51
}

中断的使用注意情况:还是把举个例子吧:
#include <msp430g2553.h>

void main(void)
{
  WDTCTL = WDTPW &#43; WDTHOLD;                 // Stop WDT
  P1DIR |= 0x01;                            // P1.0 output
  TACTL = TASSEL_2 &#43; MC_2 &#43; TAIE;           // SMCLK, contmode, interrupt  TAIE允许定时器溢出中断

  _BIS_SR(LPM0_bits &#43; GIE);                 // Enter LPM0 w/ interrupt  GIE允许中断
}

// Timer_A3 Interrupt Vector (TA0IV) handler
#pragma vector=TIMER0_A1_VECTOR
__interrupt void Timer_A(void)
{
switch( TA0IV )    //TAIV中断向量寄存器  用于
{
   case  2: break;                          // CCR1 not used   捕获/比较器1
   case  4: break;                          // CCR2 not used    捕获/比较器2
   case 10: P1OUT ^= 0x01;                  // overflow  定时器溢出
            break;
}
}



  7,注意:定时器Timer0_A的时钟可以选择为外接时钟输入TACLKP10),这样当外接一个信号时,定时器Timer0_A就相当于一个计数器使用。这样就可以用Timer0_A接外接信号,Timer1_A接标准的时钟如32768Hz的晶振,就可以实现等精度测频了。其实Timer1_A的时钟也可以外接的,但是在g2553中没有这个外接管脚(P37),所以就只能选择正常的时钟了。
Timer0_A的外接时钟输入TACLKP10)的设置如下:下面是我实现等精度测频时,两个定时器的初始化程序:
void timer0_init()
{
TACTL |= TASSEL_0&#43;MC_2&#43;TACLR;  //选择TACLK时钟作为计数时钟源,不分频   必须连续计数模式

P1SEL |= BIT0;   //P10Timer0_A的时钟TACLK输入,接外部待测信号,这样Timer0_A就当作计数器用
}
//Timer1_A采用ACLK作为时钟源计数,这样ACLK就相当于是标准信号,这样两个定时器相当于都工作在计数器方式,
//ACLK 32768Hz作为标准信号,这样可以实现等精度测频
void timer1_init()
{

TA1CCTL0 = CCIE;
TA1CCR0 = 32768;    //1s定时

TA1CTL |= TASSEL_1&#43;MC_2&#43;TACLR;  //选择ACLK时钟作为计数时钟源,不分频   必须连续计数模式

}

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-135460-1-1.html 上篇帖子: MSP430FG439上实现IO模拟UART以及移植要点解析 下篇帖子: Cisco 动态NAPT的配置(实现内部多台主机同时上网)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表