1.前期预备知识

1.1 新大陆ZigBee模块LED灯电路

CC2530学习路线-基础实验-GPIO 控制LED灯亮灭(1)-LMLPHP


从上两的两个电路可知

1.2 CC2530相关寄存器


P1 (0x90)*控制端口1的高低电平端口1.通用I/O。可以通过SFR位寻址
P1SEL (0xF4)端口1 8个子端口的功能选择P1SEL的8个bit分别代表 => P1.7~P1.0的功能选择.
值为 0:代表通用I/0(GPIO)功能.
值为 1 : 代表外设功能
P1DIR (0xFE)端口1 输入输出选择P1DIR的bit定义同P1SEL;
值为 0:代表从外部输入信号至CC2530;
值为 1:代表从CC2530输出信号至外部
P1INP (0xF6)端口1 输入模式选择P1INP定义为P1.7~P1.2的I/O输入模式。其中P1.0和P1.1是没有上拉/下拉功能
值为 0:上拉/下拉。
值为 1:三态(高电平、低电平、高阻态)
注意:端口输入模式选择中的三态分别是(高电平、低电平、高阻态),其中高阻态是让端口电阻无限大,让其外部信号改变不会影响到内部总线。

1.3 寄存器操作技巧

在官方示例文档中,推荐使用 &=~ 组合赋值运算将寄存器的某一位置为 0 ;使用 |= 组合赋值运算将寄存器的某一位置为 1 。我们可以通过以下示例来了解。

例1:将P1_0设置为GPIO功能并将设置为输出模式。
/*********官方推荐************/
// 将P1_0设置为GPIO
P1SEL &=~ 0x01;
// 将P1_0设置为输出模式
P1DIR |= 0x01; /*********一般做法**********/
P1SEL = 0x00;
P1DIR = 0x01;

使用官方推荐的操作方法和使用我们一般做法有什么好处呢?其实乍一看我们发现官方推荐方式还是比较复杂,其实一般做法比较简介,好像一般做法还更好一些。我们继续看下面的例子。

例2:假设P1SEL初始值为0x31、P1DIR初始值为0xC3.请在不改变初始值的情况下将P1_0设置为GPIO功能并将设置为输出模式。
/*********官方推荐************/

    // 将P1_0设置为GPIO
P1SEL &=~ 0x01;
// 将P1_0设置为输出模式
P1DIR |= 0x01; /*********一般做法**********/ P1SEL = 0x30;
P1DIR = 0xC3;

从例2中大家就可以发现官方推荐做法的好处,如果使用官方推荐的方式,那么只需要关注需修改的那一位,不需要知道其它位到底是怎么样的。其中最重要的就是在改变某一位的值时不会影响到其它位。

1.4 CPU空转延时

单片机实现延时的方法有很多,一般是通过执行空指令来实现延时的效果。需知道单片机的晶振频率和主频。我们这里使用的是类51单片机,使用的晶振是32MHz晶振;按照文档说法,执行以下代码可以让其延时1ms.

typedef unsigned int uint;
void delay_ms(uint ms)
{
for(uint i = 0 ; i < ms ; i ++)
{
for(uint j = 0 ; j < 535 ; j++)
}
}

1.4 操作流程图

本次实验的操作流程如下。

CC2530学习路线-基础实验-GPIO 控制LED灯亮灭(1)-LMLPHP

2.程序代码

程序代码中并无特别难的地方,根据流程图和之前的分析程序一目了然。

#include <ioCC2530.h>
#define LED1 P1_0 //P1.0端口控制LED1发光二极管
#define unint unsigned int
void init_gpio()
{
// 设置 gpio
P1SEL &= ~0x01;
// 设置 输出
P1DIR |= 0x01;
// P1端口下拉
P1 = 0;
}
void delay(unint z)
{
for (unint i = 0; i < z; i++)
{
for (unint j = 0; j < 500; j++);
}
}
void main(void) {
init_gpio();
while (1)
{
LED1 = 0; // 熄灭LED1发光二极管
delay(1000);
LED1 = 1; // 点亮LED1发光二极管
delay(1000);
}
}

The End

04-16 19:05