程序匠人--MSP430学习笔记

发布者:Wanderlust123最新更新时间:2015-10-08 来源: eefocus关键字:程序匠人  MSP430  学习笔记 手机看文章 扫描二维码
随时随地手机看文章
这只是我在学习TI 公司生产的16位超的功耗单片机MSP430的随笔,希望能对其他朋友有所借鉴,不对之处还请多指教。下面,开始430之旅。

讲解430的书现在也有很多了,不过大多数都是详细说明底层硬件结构的,看了不免有些空洞和枯燥,我认为了解一个MCU的操作首先要对其基础特性有所了解,然后再仔细研究各模块的功能。
1.首先你要知道msp430的存储器结构。典型微处理器的结构有两种:冯。诺依曼结构——程序存储器和数据存储器统一编码;哈佛结构——程序存储器和数据存储器;msp430系列单片机属于前者,而常用的mcs51系列属于后者。
0-0xf特殊功能寄存器;0x10-0x1ff外围模块寄存器;0x200-?根据不同型号地址从低向高扩展;0x1000-0x107f seg_b0x1080_0x10ff seg_a 供flash信息存储
剩下的从0xffff开始向下扩展,根据不同容量,例如149为60KB,0xffff-0x1100
2.复位信号是MCU工作的起点,430的复位型号有两种:上电复位信号POR和上电清楚信号PUC。POR信号只在上电和RST/NMI复位管脚被设置为复位功能,且低电平时系统复位。而PUC信号是POR信号产生,以及其他如看门狗定时溢出、安全键值出现错误是产生。但是,无论那种信号触发的复位,都会使msp430在地址 0xffff处读取复位中断向量,然后程序从中断向量所指的地址开始执行。复位后的状态不写了,详见参考书,嘿嘿。
3.系统时钟是一个程序运行的指挥官,时序和中断也是整个程序的核心和中轴线。430最多有三个振荡器,DCO内部振荡器;LFXT1外接低频振荡器,常见的 32768HZ,不用外接负载电容;也可接高频450KHZ-8M,需接负载电容;XT2接高频450KHZ-8M,加外接电容。
430有三种时钟信号:MCLK系统主时钟,可分频1 2 4 8,供cpu使用,其他外围模块在有选择情况下也可使用;SMCLK系统子时钟,供外围模块使用,可选则不同振荡器产生的时钟信号;ACLK辅助时钟,只能由LFXT1产生,供外围模块。
4.中断是430处理器的一大特色,因为几乎每个外围模块都能产生,430可以在没有任务时进入低功耗状态,有事件时中断唤醒cpu,处理完毕再次进入低功耗状态。
整个中断的响应过程是这样的,当有中断请求时,如果cpu处于活动状态,先完成当前命令;如果处于低功耗,先退出,将下一条指令的pc值压入堆栈;如果有多个中断请求,先响应优先级高的;执行完后,等待中断请求标志位复位,要注意,单中断源的中断请求标志位自动复位,而多中断的标志位需要软件复位;然后系统总中断允许位SR.GIE复位,相应的中断向量值装入pc,程序从这个地址继续执行。
这里要注意,中断允许位SR.GIE和中断嵌套问题。如果当你执行中断程序过程中,希望可以响应更高级别的中断请求时,必须在进入第一个中断时把 SR.GIE置位。
其实,其他的外围模块时钟沿着时钟和中断这个核心来执行的。具体的结构我也不罗索了,可以参考430系列手册。
明天开始,讲述msp430单片机C语言编程的故事。

上回把430单片机的基础特性交待了一下,让大家整体有了结构的印象,今天我想在写一下c语言对430编程的整体结构。基本上属于框架结构,即整体的模块化编程,其实这也是硬件编程的基本法则拉(可不是我规定的法则哦)。

首先是程序的头文件,包括#i nclude ,这是14系列,因为常用149;其他型号可自己修改。还可以包括#i nclude "data.h" 等数据库头文件,或函数变量声明头文件,都是你自己定义的哦。

接着就是函数和变量的声明 void Init_Sys(void);系统初始化

系统初始化是个整体的概念,广义上讲包括所有外围模块的初始化,你可以把外围模块初始化的子函数写到Init_Sys()中,也可以分别写各个模块的初始化。但结构的简洁,最好写完系统的时钟初始化后,其他所用到的模块也在这里初始化。
void Init_Sys()
{
 unsigned int i;
  BCSCTL1&=~XT2OFF;          //打开XT2振荡器
 do
 {
 IFG1 &= ~OFIFG;                  // 清除振荡器失效标志
 for (i = 0xFF; i > 0; i--);  // 延时,等待XT2起振
 }
 while ((IFG1 & OFIFG) != 0);    // 判断XT2是否起振
 BCSCTL2 =SELM_2+SELS;     //选择MCLK、SMCLK为XT2
//以下对各种模块、中断、外围设备等进行初始化
                 ........................................
  _EINT(); //打开全局中断控制
}
这里涉及到时钟问题,通常我们选择XT2为8M晶振,也即系统主时钟MCLK为8M,cpu执行命令以此时钟为准;但其他外围模块可以在相应的控制寄存器中选择其他的时钟,ACLK;当你对速度要求很低,定时时间间隔大时,就可以选择ACLK,例如在定时器Timea初始化中设置。
主程序:                       void main( void )
                                  {
                                     WDTCTL = WDTPW + WDTHOLD;              //关闭看门狗
                                    InitSys();     //初始化

                                                   //自己任务中的其他功能函数

                                         。。。。。。。。。。。。。。。。。。。。。
                                     while(1);
                                   }
主程序之后我要讲讲中断函数,中断是你做单片机任务中不可缺少的部分,也可以说是灵魂了(夸张吗)。

举个定时中断的例子:
   初始化                    void Init_Timer_A(void)
                                 {
                                  TACTL = TASSEL0 + TACLR;              // ACLK, clear TAR
                                  CCTL0 = CCIE;                         // CCR0 中断使能
                                  CCR0=32768;                           //定时1s
                                  TACTL|=MC0;                           //增计数模式
                                  }
    中断服务                #pragma vector=TIMERA0_VECTOR
                                 __interrupt void TimerA0()
                                {
                                   // 你自己要求中断执行的任务
                                 }
当然,还有其他的定时,和多种中断,各系列芯片的中断向量个数也不同。
这就是简单的整体程序框架,写得简单啦,还忘谅解,明天详细了解一下各外围模块的初始化和功能,

整体的程序设计结构,包括了所有外围模块及内部时钟,中断,定时的初始化。具体情况大家可以根据自己的需要添加或者减少,记住,模块化设计时最有力的武器。
这可是个人总结的经典阿,谢谢支持。因为经常使用149,所以这是149的结构,其他的再更改,根据个人需要。

//头文件
#i nclude
//函数声明
void InitSys();
int main( void )
{
 WDTCTL = WDTPW + WDTHOLD;              //关闭看门狗
 InitSys();     //初始化
start:
 //以下填充用户代码
LPM3;   //进入低功耗模式n,n:0~4。若不希望进入低功耗模式,屏蔽本句
goto start;
}

void InitSys()
{
 unsigned int iq0;
//使用XT2振荡器
 BCSCTL1&=~XT2OFF;          //打开XT2振荡器
 do
 {
 IFG1 &= ~OFIFG;     // 清除振荡器失效标志
 for (iq0 = 0xFF; iq0 > 0; iq0--);  // 延时,等待XT2起振
 }
 while ((IFG1 & OFIFG) != 0);    // 判断XT2是否起振
 BCSCTL2 =SELM_2+SELS;     //选择MCLK、SMCLK为XT2
//以下填充用户代码,对各种模块、中断、外围设备等进行初始化
  _EINT(); //打开全局中断控制,若不需要打开,可以屏蔽本句
}

#pragma vector=PORT2_VECTOR
__interrupt void Port2()
{
//以下为参考处理程序,不使用的端口应当删除其对于中断源的判断。
if((P2IFG&BIT0) == BIT0)
{
 //处理P2IN.0中断
 P2IFG &= ~BIT0; //清除中断标志
 //以下填充用户代码
}
else if((P2IFG&BIT1) ==BIT1)
{
 //处理P2IN.1中断
 P2IFG &= ~BIT1; //清除中断标志
 //以下填充用户代码
}
else if((P2IFG&BIT2) ==BIT2)
{
 //处理P2IN.2中断
 P2IFG &= ~BIT2; //清除中断标志
 //以下填充用户代码
}
else if((P2IFG&BIT3) ==BIT3)
{
 //处理P2IN.3中断
 P2IFG &= ~BIT3; //清除中断标志
 //以下填充用户代码
}
else if((P2IFG&BIT4) ==BIT4)
{
 //处理P2IN.4中断
 P2IFG &= ~BIT4; //清除中断标志
 //以下填充用户代码
}
else if((P2IFG&BIT5) ==BIT5)
{
 //处理P2IN.5中断
 P2IFG &= ~BIT5; //清除中断标志
 //以下填充用户代码
}
else if((P2IFG&BIT6) ==BIT6)
{
 //处理P2IN.6中断
 P2IFG &= ~BIT6; //清除中断标志
 //以下填充用户代码
}
else
{
 //处理P2IN.7中断
 P2IFG &= ~BIT7; //清除中断标志

//以下填充用户代码
}

LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=USART1TX_VECTOR
__interrupt void Usart1Tx()
{
//以下填充用户代码
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=USART1RX_VECTOR
__interrupt void Ustra1Rx()
{
//以下填充用户代码
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=PORT1_VECTOR
__interrupt void Port1()
{
//以下为参考处理程序,不使用的端口应当删除其对于中断源的判断。
if((P1IFG&BIT0) == BIT0)
{
 //处理P1IN.0中断
 P1IFG &= ~BIT0; //清除中断标志
 //以下填充用户代码
}
else if((P1IFG&BIT1) ==BIT1)
{
 //处理P1IN.1中断
 P1IFG &= ~BIT1; //清除中断标志
//以下填充用户代码
}
else if((P1IFG&BIT2) ==BIT2)
{
 //处理P1IN.2中断
 P1IFG &= ~BIT2; //清除中断标志
 //以下填充用户代码
}
else if((P1IFG&BIT3) ==BIT3)
{
 //处理P1IN.3中断
 P1IFG &= ~BIT3; //清除中断标志
 //以下填充用户代码
}
else if((P1IFG&BIT4) ==BIT4)
{
 //处理P1IN.4中断
 P1IFG &= ~BIT4; //清除中断标志
 //以下填充用户代码
}
else if((P1IFG&BIT5) ==BIT5)
{
 //处理P1IN.5中断
 P1IFG &= ~BIT5; //清除中断标志
 //以下填充用户代码
}
else if((P1IFG&BIT6) ==BIT6)
{
 //处理P1IN.6中断
 P1IFG &= ~BIT6; //清除中断标志
 //以下填充用户代码
}
else
{
 //处理P1IN.7中断
 P1IFG &= ~BIT7; //清除中断标志
 //以下填充用户代码
}
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}[page]

#pragma vector=TIMERA1_VECTOR
__interrupt void TimerA1()
{
//以下为参考处理程序,不使用的中断源应当删除
switch (__even_in_range(TAIV, 10))
{
 case 2:
 //捕获/比较1中断
 //以下填充用户代码
 break;
 case 4:
 //捕获/比较2中断
 //以下填充用户代码
 break;
 case 10:
 //TAIFG定时器溢出中断
 //以下填充用户代码
 break;
}
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=TIMERA0_VECTOR
__interrupt void TimerA0()
{
//以下填充用户代码
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=ADC_VECTOR
__interrupt void Adc()
{
//以下为参考处理程序,不使用的中断源应当删除
if((ADC12IFG&BIT0)==BIT0)
{
 //通道0
 //以下填充用户代码
}
else if((ADC12IFG&BIT1)==BIT1)
{
 //通道1
 //以下填充用户代码
}
else if((ADC12IFG&BIT2)==BIT2)
{
//通道2
 //以下填充用户代码
}
else if((ADC12IFG&BIT3)==BIT3)
{
 //通道3
 //以下填充用户代码
}
else if((ADC12IFG&BIT4)==BIT4)
{
 //通道4
 //以下填充用户代码
}
else if((ADC12IFG&BIT5)==BIT5)
{
 //通道5
//以下填充用户代码
}
else if((ADC12IFG&BIT6)==BIT6)
{
 //通道6
 //以下填充用户代码
}
else if((ADC12IFG&BIT7)==BIT7)
{
 //通道7
 //以下填充用户代码
}
else if((ADC12IFG&BIT8)==BIT8)
{
 //VeREF+
 //以下填充用户代码
}
else if((ADC12IFG&BIT9)==BIT9)
{
 //VREF-/VeREF-
 //以下填充用户代码
}
else if((ADC12IFG&BITA)==BITA)
{
 //温度
 //以下填充用户代码
}
else if((ADC12IFG&BITB)==BITB)
{
 //(AVcc-AVss)/2
 //以下填充用户代码
}
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=USART0TX_VECTOR
__interrupt void Usart0Tx()
{
//以下填充用户代码
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=USART0RX_VECTOR
__interrupt void Usart0Rx()
{
//以下填充用户代码
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=WDT_VECTOR
__interrupt void WatchDog()
{
//以下填充用户代码
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=COMPARATORA_VECTOR
__interrupt void ComparatorA()
{
//以下填充用户代码
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=TIMERB1_VECTOR
__interrupt void TimerB1()
{
//以下为参考处理程序,不使用的中断源应当删除
switch (__even_in_range(TBIV, 14))
{
 case 2:
 //捕获/比较1中断
 //以下填充用户代码
 break;
 case 4:
 //捕获/比较2中断
 //以下填充用户代码
 break;
 case 6:
 //捕获/比较3中断
 //以下填充用户代码
 break;
 case 8:
//捕获/比较4中断
 //以下填充用户代码
 break;
 case 10:
 //捕获/比较5中断
 //以下填充用户代码
 break;
 case 12:
 //捕获/比较6中断
 //以下填充用户代码
 break;
 case 14:
 //TBIFG定时器溢出中断
 //以下填充用户代码
 break;
}
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=TIMERB0_VECTOR
__interrupt void TimerB0()
{
//以下填充用户代码
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=NMI_VECTOR
__interrupt void Nmi()
{
//以下为参考处理程序,不使用的中断源应当删除
if((IFG1&OFIFG)==OFIFG)
{
 //振荡器失效
 IFG1 &= ~OFIFG;
 //以下填充用户代码
}
else if((IFG1&NMIIFG)==NMIIFG)
{
 //RST/NMI不可屏蔽中断
 IFG1 &= ~NMIIFG;
 //以下填充用户代码
}
else //if((FCTL3&ACCVIFG)==ACCVIFG)
{
 //存储器非法访问
 FCTL3 &= ~ACCVIFG;
 //以下填充用户代码
}
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

#pragma vector=BASICTIMER_VECTOR
__interrupt void BasTimer()
{
//以下填充用户代码
LPM3_EXIT; //退出中断后退出低功耗模式。若退出中断后要保留低功耗模式,将本句屏蔽
}

关键字:程序匠人  MSP430  学习笔记 引用地址:程序匠人--MSP430学习笔记

上一篇:MSP430开发注意事项
下一篇:MSP430的头文件解析

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

MSP432P401R TI Drivers 库函数学习笔记(五)PWM
平台:Code Composer Studio 10.4.0 MSP432P401R SimpleLink™ 微控制器 LaunchPad™ 开发套件 (MSP-EXP432P401R) API (机翻) PWM API 官方手册 函数 void PWM_close (PWM_Handle handle) 函数关闭由PWM句柄指定的PWM实例 int_fast16_t PWM_control (PWM_Handle handle, uint_fast16_t cmd, void *arg) 函数在给定的PWM_Handle上执行特定的实现特性 void PWM_init (void) 这个函数初始化PW
[单片机]
MSP432P401R TI Drivers 库函数<font color='red'>学习</font><font color='red'>笔记</font>(五)PWM
MSP430单片机AD转换LCD1602&TUBE显示
#include msp430x14x.h #define uint unsigned int #define uchar unsigned char #define ulint unsigned long int #define RS BIT0; #define RW BIT1; #define EN BIT2; uint Volt0; //设置电压变量 ulint Volttem0; unsigned data0=0,data1=0; uint ADresult0; //设置A/D转换结果 uint a ={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,
[单片机]
MSP430 flash写入(一)
/* --COPYRIGHT--,BSD_EX * Copyright (c) 2012, Texas Instruments Incorporated * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * * Redistributions of source code must retain the above c
[单片机]
MSP430数字滤波器设计分享
大家一直有在讨论使用TI内部的数字滤波器,但是由于其调用的麻烦,以及汇编代码看的累,所以经常会碰到各式各样的问题,我以前也有帮工程师解惑,现在做个分享帖给大家,希望大家多多支持,让MSP430的作用更为广泛。 1.软件介绍:Filter solution 2.软件界面介绍: 该软件不仅能设计数字滤波器,模拟滤波器一样可以搞定。 3.开始设计数字滤波器: 4.频响测试: 5.代码生成:
[单片机]
<font color='red'>MSP430</font>数字滤波器设计分享
基于MSP430与uPD720200的高速温度采集系统的设计
0 引言 USB3.0高速数据采集系统,以其5Gbps的带宽和即插即用等优越的性能,越来越受到人们的重视。目前常用的USB3.0通信芯片主要有来自美国赛普拉斯(CYPRESS)公司开发的EZ USB FX3系列芯片和日本NEC公司开发的uPD720200系列芯片。二者不仅都具有集成度高、功能强大、兼容USB2.0等优点,而且两个厂家都提供了功能强大的开发工具包,能极大缩短开发周期,而深受开发者喜爱。目前一些采集系统采用传统的51系列单片机为主控芯片,则不能充分发挥USB3.0的带宽,性能不佳。虽然CPLD芯片可以工作在很高的频率,但不及单片机灵活易用,故本文采用美国TI公司最新高速的ADS7886芯片为A/D转换芯片,最新的MSP4
[单片机]
基于<font color='red'>MSP430</font>与uPD720200的高速温度采集系统的设计
MSP430单片机的OLED实时时钟制作
这段时间在学习MSP430,将以前学习的常用模块移植过来,由于MSP430F149没有iic接口,自己模拟了一个,用到的模块为:DS1302、DS18B20、OLED(IIC接口)。可通过按键切换AD及温度显示(红外程序也包含其中,有兴趣的朋友可以稍微改一下就能实现红外控制) 制作出来的实物图如下: 温度 AD #define main_c #include msp430x14x.h #include main.h #include iic.h #include ds1302.h #include ds18b20.h #include adc12.h #include IrRed.h #include
[单片机]
<font color='red'>MSP430</font>单片机的OLED实时时钟制作
MSP430 IIC 协议程序
坐在电脑前写了一整个下午加一个晚上,终于在430上完成了 IIC 协议,在此附上 代码吧,留个纪念,睡觉去,晚安! #ifndef __IIC_h #define __IIC_h #include basic.h //--------------------------------------- //宏定义 #define IIC_OUT P1OUT #define IIC_DIR P1DIR #define IIC_SEL P1SEL #define IIC_IN P1IN #define IIC_SDA_IN IIC_IN&0X01//取SDA读取的值,根据接的高地位不同,与上不同的BIT #define SDA_OU
[单片机]
MSP430用按键中断控制定时器产生持续 1s 的周期信号
////////////////////////////////////////////////////////////////////////////////////////////////////// // 2012 年吉林省大学生电子设计大赛 C题 声源定位系统 ----(声源部分) // 要求: 有信号产生电路, 放大电路, 声音频率 800Hz 左右, 按一次键响声持续 1 秒. // 声源模块体积不超过 4cm*4cm*4cm. //==========================================================================================
[单片机]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

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

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

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