一、头文件:

#ifndef __MK60_GPIO_H__
#define __MK60_GPIO_H__ #include "MK60_gpio_cfg.h" /*
* 定义管脚方向
*/
typedef enum GPIO_CFG
{
//这里的值不能改!!!
GPI = 0, //定义管脚输入方向 GPIOx_PDDRn里,0表示输入,1表示输出
GPO = 1, //定义管脚输出方向
} GPIO_CFG; #define HIGH 1u
#define LOW 0u extern GPIO_MemMapPtr GPIOX[PTX_MAX];
#define GPIOX_BASE(PTxn) GPIOX[PTX(PTxn)] //GPIO模块的地址 /****************************外部使用****************************/ extern void gpio_init (PTXn_e, GPIO_CFG, uint8 data); //初始化gpio
extern void gpio_ddr (PTXn_e, GPIO_CFG); //设置引脚数据方向
extern void gpio_set (PTXn_e, uint8 data); //设置引脚状态
extern void gpio_turn (PTXn_e); //反转引脚状态
extern uint8 gpio_get (PTXn_e); //读取引脚状态 //如下 4个 函数 的 PTxn 只能是 宏定义,不能是 变量
#define GPIO_SET(PTxn,data) (PTXn_T(PTxn,OUT)= (data)) //设置输出电平
#define GPIO_TURN(PTxn) (PTXn_T(PTxn,T)= 1) //翻转输出电平
#define GPIO_GET(PTxn) (PTXn_T(PTxn,IN)) //读取引脚输入状态
#define GPIO_DDR(PTxn,ddr) (PTXn_T(PTxn,DDR) = ddr) //输入输出状态 //如下 函数 的 PTxn 可以是 宏定义,也可以是 变量 //n位操作
#define GPIO_SET_NBIT(NBIT,PTxn,data) GPIO_PDOR_REG(GPIOX_BASE(PTxn)) = ( \
( \
GPIO_PDOR_REG(GPIOX_BASE(PTxn)) \
& \
((uint32)( ~(((1<<NBIT)-1)<<PTn(PTxn)))) \
) \
| ((data) \
&( \
((1<<NBIT)-1) \
<<PTn(PTxn) \
)) \
) #define GPIO_DDR_NBIT(NBIT,PTxn,ddr) GPIO_PDDR_REG(GPIOX_BASE(PTxn)) = ( \
( \
GPIO_PDDR_REG(GPIOX_BASE(PTxn)) \
& \
((uint32)( ~(((1<<NBIT)-1)<<PTn(PTxn)))) \
) \
| ((ddr) \
&( \
((1<<NBIT)-1) \
<<PTn(PTxn) \
)) \
) #define GPIO_T_NBIT(NBIT,PTxn,data) GPIO_PTOR_REG(GPIOX_BASE(PTxn)) = ( \
( \
GPIO_PTOR_REG(GPIOX_BASE(PTxn)) \
& \
((uint32)( ~(((1<<NBIT)-1)<<PTn(PTxn)))) \
) \
| ((data) \
&( \
((1<<NBIT)-1) \
<<PTn(PTxn) \
)) \
) #define GPIO_GET_NBIT(NBIT,PTxn) (( GPIO_PDIR_REG(GPIOX_BASE(PTxn))>>PTn(PTxn) ) & ((1<<NBIT)-1)) #endif //__MK60_GPIO_H__

二、源文件:

/*
* 包含头文件
*/
#include "common.h"
#include "MK60_port.h"
#include "MK60_gpio.h" /*
* 定义数组
*/
GPIO_MemMapPtr GPIOX[PTX_MAX] = {PTA_BASE_PTR, PTB_BASE_PTR, PTC_BASE_PTR, PTD_BASE_PTR, PTE_BASE_PTR}; //定义五个指针数组保存 GPIOX 的地址 /*!
* @brief 初始化gpio
* @param PTxn 端口
* @param cfg 引脚方向,0=输入,1=输出
* @param data 输出初始状态,0=低电平,1=高电平 (对输入无效)
* @since v5.0
* Sample usage: gpio_init (PTA8, GPI,0); //初始化 PTA8 管脚为输入
*/
void gpio_init (PTXn_e ptxn, GPIO_CFG cfg, uint8 data)
{
//复用管脚为GPIO功能
port_init( ptxn, ALT1); //端口方向控制输入还是输出
if( cfg == GPI )
{
//设置端口方向为输入
GPIO_PDDR_REG(GPIOX_BASE(ptxn)) &= ~(1 << PTn(ptxn)); // GPIO PDDR 管脚号 清0,即对应管脚配置为端口方向输入
}
else
{
//设置端口方向为输出
GPIO_PDDR_REG(GPIOX_BASE(ptxn)) |= (1 << PTn(ptxn)); // GPIO PDDR 管脚号 置1,即对应管脚配置为端口方向输出 //端口输出数据
if(data == 0)
{
GPIO_PDOR_REG(GPIOX_BASE(ptxn)) &= ~(1 << PTn(ptxn)); // GPIO PDOR 管脚号 清0,即对应管脚配置为端口输出低电平
}
else
{
GPIO_PDOR_REG(GPIOX_BASE(ptxn)) |= (1 << PTn(ptxn)); // GPIO PDOR 管脚号 置1,即对应管脚配置为端口输出高电平
}
}
} /*!
* @brief 设置引脚数据方向
* @param PTxn 端口
* @param cfg 引脚方向,0=输入,1=输出
* @since v5.0
* Sample usage: gpio_ddr (PTA8, GPI); //设置 PTA8 管脚为输入
*/
void gpio_ddr (PTXn_e ptxn, GPIO_CFG cfg)
{
//端口方向控制输入还是输出
if( cfg == GPI )
{
//设置端口方向为输入
GPIO_PDDR_REG(GPIOX_BASE(ptxn)) &= ~(1 << PTn(ptxn)); // GPIO PDDR 管脚号 清0,即对应管脚配置为端口方向输入
}
else
{
//设置端口方向为输出
GPIO_PDDR_REG(GPIOX_BASE(ptxn)) |= (1 << PTn(ptxn)); // GPIO PDDR 管脚号 置1,即对应管脚配置为端口方向输出
}
} /*!
* @brief 设置引脚状态
* @param PTxn 端口
* @param data 输出初始状态,0=低电平,1=高电平 (对输入无效)
* @since v5.0
* @warning 务必保证数据方向为输出(DEBUG模式下,有断言进行检测)
* Sample usage: gpio_set (PTA8, 1); // PTA8 管脚 输出 1
*/
void gpio_set (PTXn_e ptxn, uint8 data)
{
ASSERT( BIT_GET( GPIO_PDDR_REG(GPIOX_BASE(ptxn)) , PTn(ptxn)) == GPO ); // 断言,检测 输出方向是否为输出
// 获取 GPIO PDDR 管脚号 ,比较是否为输出 //端口输出数据
if(data == 0)
{
GPIO_PDOR_REG(GPIOX_BASE(ptxn)) &= ~(1 << PTn(ptxn)); // GPIO PDOR 管脚号 清0,即对应管脚配置为端口输出低电平
}
else
{
GPIO_PDOR_REG(GPIOX_BASE(ptxn)) |= (1 << PTn(ptxn)); // GPIO PDOR 管脚号 置1,即对应管脚配置为端口输出高电平
}
} /*!
* @brief 反转引脚状态
* @param PTxn 端口
* @since v5.0
* @warning 务必保证数据方向为输出(DEBUG模式下,有断言进行检测)
* Sample usage: gpio_turn (PTA8); // PTA8 管脚 输出 反转
*/
void gpio_turn (PTXn_e ptxn)
{
ASSERT( BIT_GET( GPIO_PDDR_REG(GPIOX_BASE(ptxn)) , PTn(ptxn)) == GPO ); // 断言,检测 输出方向是否为输出
// 获取 GPIO PDDR 管脚号 ,比较是否为输出 GPIO_PTOR_REG( GPIOX_BASE(ptxn)) = 1 << (PTn(ptxn )); // GPIO PTOR ptxn 置1,其他清0 ,即对应管脚配置为端口输出反转,其他位不变
// 此处不能用 BIT_SET 这个宏来置1 ,因为必须保证其他位 不变,其他位直接清0即可
} /*!
* @brief 读取引脚输入状态
* @param PTxn 端口
* @return 管脚的状态,1为高电平,0为低电平
* @since v5.0
* @warning 务必保证数据方向为输入(DEBUG模式下,有断言进行检测)
* Sample usage: uint8 pta8_data = gpio_get (PTA8); // 获取 PTA8 管脚 输入电平
*/
uint8 gpio_get(PTXn_e ptxn)
{
ASSERT( BIT_GET( GPIO_PDDR_REG(GPIOX_BASE(ptxn)) , PTn(ptxn)) == GPI ); // 断言,检测 输出方向是否为输入
// 获取 GPIO PDDR 管脚号 ,比较是否为输入 return ((GPIO_PDIR_REG(GPIOX_BASE(ptxn)) >> PTn(ptxn )) & 0x01); // 获取 GPIO PDIR ptxn 状态,即读取管脚输入电平
}

 

 

 

 

三、小结:

    主要是学会怎么调用,出现问题的话,检查寄存器的配置(不过一般不需要)。

智能车学习(十四)——K60单片机GPIO学习的更多相关文章

  1. 深度学习课程笔记(十四)深度强化学习 --- Proximal Policy Optimization (PPO)

    深度学习课程笔记(十四)深度强化学习 ---  Proximal Policy Optimization (PPO) 2018-07-17 16:54:51  Reference: https://b ...

  2. scrapy爬虫学习系列四:portia的学习入门

    系列文章列表: scrapy爬虫学习系列一:scrapy爬虫环境的准备:      http://www.cnblogs.com/zhaojiedi1992/p/zhaojiedi_python_00 ...

  3. 智能车学习(十五)——K60野火2013版例程

    一.中断函数注册方法: 1.格式: 配置某个功能的中断 注册中断函数 开启中断 2.一个例子 pit_init_ms(PIT0,);//定时中断初始化 set_vector_handler(PIT0_ ...

  4. 《SLAM十四讲》个人学习知识点梳理

    0.引言 从六月末到八月初大概一个月时间一直在啃SLAM十四讲[1]这本书,这本书把SLAM中涉及的基本知识点都涵盖了,所以在这里做一个复习,对这本书自己学到的东西做一个梳理. 书本地址:http:/ ...

  5. 强化学习(十四) Actor-Critic

    在强化学习(十三) 策略梯度(Policy Gradient)中,我们讲到了基于策略(Policy Based)的强化学习方法的基本思路,并讨论了蒙特卡罗策略梯度reinforce算法.但是由于该算法 ...

  6. Scala学习十四——模式匹配和样例类

    一.本章要点 match表达式是更好的switch,不会有意外调入下一个分支 如果没有模式能够匹配,会抛出MatchError,可以用case _模式避免 模式可以包含一个随意定义的条件,称做守卫 你 ...

  7. 强化学习(十九) AlphaGo Zero强化学习原理

    在强化学习(十八) 基于模拟的搜索与蒙特卡罗树搜索(MCTS)中,我们讨论了MCTS的原理和在棋类中的基本应用.这里我们在前一节MCTS的基础上,讨论下DeepMind的AlphaGo Zero强化学 ...

  8. JMeter学习(十四)JMeter函数学习(转载)

    转载自 http://www.cnblogs.com/yangxia-test JMeter函数是一些能够转化在测试树中取样器或者其他配置元件的域的特殊值.一个函数的调用就像这样:${_functio ...

  9. android学习十四(android的接收短信)

    收发短信是每一个手机主要的操作,android手机当然也能够接收短信了. android系统提供了一系列的API,使得我们能够在自己的应用程序里接收和发送短信. 事实上接收短信主要是利用我们前面学过的 ...

随机推荐

  1. Spring4 与 Hibernate4 整合过程中的问题记录

    Spring4使用注解配置,很方便也很有趣,就是有一些坑需要自己去发现和解决,笔者列出自己在使用过程中遇到的问题,希望对您有所帮助. 1.如果使用hibernate.cfg.xml配置文件配置Hibe ...

  2. Effective C++ -----条款38:通过复合塑模出has-a或“根据某物实现出”

    复合(composition)的意义和public继承完全不同. 在应用域(application domain),复合意味has-a(有一个).在实现域(implementation domain) ...

  3. nyoj221_Tree_subsequent_traversal

    Tree 时间限制:1000 ms  |  内存限制:65535 KB 难度:3   描述 Little Valentine liked playing with binary trees very ...

  4. CString之GetBuffer与ReleaseBuffer

    我们知道,CString是MFC中提供的方便字符串操作的一个类,非常好使,具有自动动态内存管理功能. GetBuffer()主要作用是将字符串的缓冲区长度锁定: ReleaseBuffer()则是解除 ...

  5. log4net使用

    平时项目里一直都有在使用log4net作为日志记录模块,当时一直都没有去理解log4net的配置文件信息.今天抽出了一点时间来看了看配置文件信息. log4net配置文件信息: <log4net ...

  6. 从下拉菜单拖拽一个元素 出来,插入到页面中的app 列表中

    1,实现功能:从下拉菜单拖拽一个元素 出来,插入到页面中的app 列表中 并实现app向后移动一个元素的位置: 2.实现思路: 01.遍历下拉菜单,添加拖拽方法,实现位置移动功能: 02.遍历app列 ...

  7. 自定义Button 的图片设置不显示问题。

    如果你是自定义button  那么你设置图片就要用 button.imageView.image = [UIImage imageName:@""]; 如果你是给系统原生的butt ...

  8. php数据访问(修改)

    修改:跟添加相似,需要显示默认值 先嵌入php代码  查询数据库 $code = $_GET["c"]; $db = new MySQLi("localhost" ...

  9. ramnit病毒

    今天打开Hbuilder,发现每个html文档都有如下代码: <SCRIPT Language=VBScript><!--DropFileName = "svchost.e ...

  10. eclipse maven tomcat7 热部署

    .配置tomcat a.配置jdk b.CATALINA_HOME=c:\tomcat CATALINA_BASE=c:\tomcat .tomcat配置密码 C:\Program Files\oth ...