新闻中心

EEPW首页>嵌入式系统>设计应用> MSP430(f149)学习笔记——红外遥控发射

MSP430(f149)学习笔记——红外遥控发射

作者: 时间:2016-11-28 来源:网络 收藏
最近由于项目需求,研究了一下MSP430,总体感觉MSP430还是非常不错的单片机,不亏是TI的产品。项目中用MSP430实现了红外遥控功能,下面我们先来分析一下红外的发送原理。
红外发送原理

红外发送实际上是按照一定的时间间隔,断断续续的发送红外射线来进行数据传输,为了提高数据传输的准确定和降低功耗,红外一般是采用38K的红外调制信号,其中,38K允许上下浮动1K以内的误差。红外发送的开头需要一个引导位,引导位是发送9ms,然后停止4.5ms,如果发送完一个数据之后还需要再发数据的话还需要连发码,连发码是先发送9ms,然后停止4.5ms。引导码和连发码的波形如下:

本文引用地址://m.amcfsurvey.com/article/201611/322815.htm

引导码和连发码之后是32位数据位,这32位数据位实际上是一个字节,其中,前16位是用户自定义地址码,用来区分红外是发给谁的,比如,电视的红外遥控是不能操作冰箱或空调的,剩下的16位是传输的数据,其中前8位是要传送的一个字节,后8位是这个字节的反码,用来校验数据传输的是否正确的,数据位无非就是0或1,其中0是先发送560us红外,然后停止565us,1是先发送560us然后停止1685us,0和1的波形如下:

每次传输实际上就是一个引导码加32位数据位或者一个连发码加32位数据位,整体的波形如下:



注意,你上所有的波形都是接收时的波形,由于红外接收头,如HS0038B,在采集到38K红外线时输出低电平,在采集不到38K红外线时输出高电平,所以,当发送红外线时在波形中为低电平,停止发送时反而为高电平。

实现电路
熟习了原理以后,再来设计电路图就非常简单了,用两个三极管停止串联,其中一个的基极(无论是PNP还是NPN,都是旁边那个脚)接到38K方波上,另一个的基极接到发送控制引脚上,我这里没有自己来绘制原理图,采用了一个网上的原理图,具体如下:

代码设计

下面的原理部分都理通了,下面,我们用MSP430来实现设计,其中P2.3口是MSP430的PWM输出引脚,这里是输出38K方波,P2.2口为发送停止控制引脚。具体代码如下:

  1. #include<msp430x14x.h>
  2. staticunsignedintcnt=0;
  3. voiddelay_ms(unsignedintms){
  4. cnt=ms;
  5. TBCCR0=1000;
  6. TBCTL=CNTL_0+TASSEL_2+MC_1+ID_3;
  7. TBCCTL0=CCIE;
  8. while(cnt!=0);
  9. TBCTL=MC_0;
  10. }
  11. voiddelay_us(unsignedintus){
  12. cnt=1;
  13. TBCCR0=us;
  14. TBCTL=CNTL_0+TASSEL_2+MC_1+ID_3;
  15. TBCCTL0=CCIE;
  16. while(cnt!=0);
  17. TBCTL=MC_0;
  18. }
  19. voidir_open(){
  20. P2DIR|=BIT2|BIT3;//P2.2,P2.3输出
  21. P2SEL|=BIT3;//P2.2:IOP2.3:TA0
  22. P2OUT&=~(BIT2|BIT3);
  23. //38K->P2.3
  24. CCR0=(int)(26.3*8+0.5);
  25. CCTL1=OUTMOD_6;
  26. CCR1=(int)(13.15*8+0.5);
  27. TACTL=TASSEL_2+MC_1;
  28. }
  29. voidir_start(){
  30. P2OUT|=BIT2;
  31. delay_us(9000);
  32. P2OUT&=~BIT2;
  33. delay_us(4500);
  34. }
  35. voidir_next(){
  36. P2OUT|=BIT2;
  37. delay_us(9000);
  38. P2OUT&=~BIT2;
  39. delay_us(2250);
  40. }
  41. voidir_send_byte(unsignedcharc){
  42. unsignedchari;
  43. for(i=0;i!=8;++i){
  44. P2OUT|=BIT2;
  45. delay_us(560);
  46. P2OUT&=~BIT2;
  47. if(c&0x01){
  48. delay_us(1685);
  49. }
  50. else{
  51. delay_us(565);
  52. }
  53. c>>=1;
  54. }
  55. }
  56. voidir_end(){
  57. P2OUT|=BIT2;
  58. delay_us(300);
  59. P2OUT&=~BIT2;
  60. }
  61. voidir_put(unsignedcharc){
  62. ir_start();
  63. ir_send_byte(0x00);
  64. ir_send_byte(0xff);
  65. ir_send_byte(c);
  66. ir_send_byte(~c);
  67. ir_end();
  68. }
  69. voidir_close(){
  70. P2SEL&-~BIT3;
  71. P2DIR|=BIT3;
  72. P2OUT&=~BIT3;
  73. TACTL=TACLR;
  74. }
  75. voidmain(void)
  76. {
  77. WDTCTL=WDTPW+WDTHOLD;//StopWDT
  78. BCSCTL1&=~XT2OFF;//使能XT2
  79. do
  80. {
  81. IFG1&=~OFIFG;//清除XT2标志
  82. _NOP();
  83. }
  84. while((IFG1&OFIFG));//等待外部晶振稳定即寄存器IFGI的OFIFG位等于0
  85. BCSCTL2|=SELM_2;//将MCLK配置为XT2
  86. BCSCTL2|=SELS;//将SMCLK配置1/2XT2即4M
  87. _EINT();
  88. ir_open();
  89. //ir_close();//关闭红外函数,这里不用
  90. while(1){
  91. delay_ms(1000);
  92. //P2OUT^=BIT2;
  93. ir_put(0x83);
  94. }
  95. }
  96. #pragmavector=TIMERB0_VECTOR
  97. __interruptvoidtimerb_handler(){
  98. --cnt;
  99. }

好了,看过上面这些资料,红外的发送就不会有什么疑问了,调试的时候我是用51单片机的红外解码测试程序来调试的,现在已经完全没有问题了,如果有什么疑问,欢迎留言讨论。


评论


技术专区

关闭