stm32 cubemx ps2无线(有线)手柄

这篇具有很好参考价值的文章主要介绍了stm32 cubemx ps2无线(有线)手柄。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。


前言

本文讲解使用cubemx配置PS2手柄实现对手柄的按键和模拟值的读取。
很简单,库已经封装好了,直接就可以了。

文件
stm32 cubemx ps2无线(有线)手柄,stm32,嵌入式硬件,单片机


一、cubemx配置

这个很简单,不需要走协议,只要配置gpio就可以了
stm32 cubemx ps2无线(有线)手柄,stm32,嵌入式硬件,单片机

二、代码

1.引入库

使用我的两个库

bsp_hal_ps2.c

/* 包含头文件 ----------------------------------------------------------------*/
#include "bsp_hal_ps2.h"

/* 私有类型定义 --------------------------------------------------------------*/

/* 私有宏定义 ----------------------------------------------------------------*/
#define DI()    HAL_GPIO_ReadPin(PS2_DI_GPIO_Port, PS2_DI_Pin)           //PB12  输入

#define DO_H()  HAL_GPIO_WritePin(PS2_DO_CMD_GPIO_Port, PS2_DO_CMD_Pin, GPIO_PIN_SET)        //命令位高
#define DO_L()  HAL_GPIO_WritePin(PS2_DO_CMD_GPIO_Port, PS2_DO_CMD_Pin, GPIO_PIN_RESET)        //命令位低

#define CS_H()  HAL_GPIO_WritePin(PS2_CS_SEL_GPIO_Port, PS2_CS_SEL_Pin, GPIO_PIN_SET)       //CS拉高
#define CS_L()  HAL_GPIO_WritePin(PS2_CS_SEL_GPIO_Port, PS2_CS_SEL_Pin, GPIO_PIN_RESET)       //CS拉低

#define CLK_H() HAL_GPIO_WritePin(PS2_CLK_GPIO_Port, PS2_CLK_Pin, GPIO_PIN_SET)      //时钟拉高
#define CLK_L() HAL_GPIO_WritePin(PS2_CLK_GPIO_Port, PS2_CLK_Pin, GPIO_PIN_RESET)      //时钟拉低
/* 私有变量 ------------------------------------------------------------------*/
const uint8_t Comd[2]={0x01,0x42};	//开始命令。请求数据
const uint16_t MASK[16] = {
    PSB_SELECT,
    PSB_L3,
    PSB_R3 ,
    PSB_START,
    PSB_PAD_UP,
    PSB_PAD_RIGHT,
    PSB_PAD_DOWN,
    PSB_PAD_LEFT,
    PSB_L2,
    PSB_R2,
    PSB_L1,
    PSB_R1 ,
    PSB_GREEN,
    PSB_RED,
    PSB_BLUE,
    PSB_PINK
};	//按键值与按键明
/* 扩展变量 ------------------------------------------------------------------*/
_u_PS2_Data PS2_Data={0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00}; //数据存储数组
/* 私有函数原形 --------------------------------------------------------------*/

/* 函数体 --------------------------------------------------------------------*/
static void delay_us(int16_t us)
{
	for(int i=0; i<us*10; i++)
	{
			;
	}
}
//向手柄发送命令
static uint8_t PS2_Cmd(uint8_t CMD)
{
	volatile uint16_t ref=0x01;
	uint8_t Data;

	Data = 0;
	for(ref=0x01; ref<0x0100; ref<<=1)
	{
		if(ref&CMD)
		{
			DO_H();                   //输出以为控制位
		}
		else DO_L();

		CLK_H();                        //时钟拉高
		delay_us(10);
		CLK_L();
		delay_us(10);
		CLK_H();
		if(DI())
		{
			Data = ref|Data;
		}
	}
	
	return Data;
}
//short poll
static void PS2_ShortPoll(void)
{
	CS_L();
	delay_us(16);
	PS2_Cmd(0x01);  
	PS2_Cmd(0x42);  
	PS2_Cmd(0X00);
	PS2_Cmd(0x00);
	PS2_Cmd(0x00);
	CS_H();
	delay_us(16);	
}

//进入配置
static void PS2_EnterConfing(void)
{
    CS_L();
	delay_us(16);
	PS2_Cmd(0x01);  
	PS2_Cmd(0x43);  
	PS2_Cmd(0X00);
	PS2_Cmd(0x01);
	PS2_Cmd(0x00);
	PS2_Cmd(0X00);
	PS2_Cmd(0X00);
	PS2_Cmd(0X00);
	PS2_Cmd(0X00);
	CS_H();
	delay_us(16);
}

//发送模式设置
static void PS2_TurnOnAnalogMode(void)
{
	CS_L();
	PS2_Cmd(0x01);  
	PS2_Cmd(0x44);  
	PS2_Cmd(0X00);
	PS2_Cmd(0x01); //analog=0x01;digital=0x00  软件设置发送模式
	PS2_Cmd(0xEE); //Ox03锁存设置,即不可通过按键“MODE”设置模式。
				   //0xEE不锁存软件设置,可通过按键“MODE”设置模式。
	PS2_Cmd(0X00);
	PS2_Cmd(0X00);
	PS2_Cmd(0X00);
	PS2_Cmd(0X00);
	CS_H();
	delay_us(16);
}
//振动设置
static void PS2_VibrationMode(void)
{
	CS_L();
	delay_us(16);
	PS2_Cmd(0x01);  
	PS2_Cmd(0x4D);  
	PS2_Cmd(0X00);
	PS2_Cmd(0x00);
	PS2_Cmd(0X01);
	CS_H();
	delay_us(16);	
}
//完成并保存配置
static void PS2_ExitConfing(void)
{
    CS_L();
	delay_us(16);
	PS2_Cmd(0x01);  
	PS2_Cmd(0x43);  
	PS2_Cmd(0X00);
	PS2_Cmd(0x00);
	PS2_Cmd(0x5A);
	PS2_Cmd(0x5A);
	PS2_Cmd(0x5A);
	PS2_Cmd(0x5A);
	PS2_Cmd(0x5A);
	CS_H();
	delay_us(16);
}
//清除数据缓冲区
void PS2_ClearData(void)
{
	PS2_Data.DATE.ID = 0;
	PS2_Data.DATE.verify = 0;
	PS2_Data.DATE.key1_U.byte = 0xff;
	PS2_Data.DATE.key2_U.byte = 0xff;
	PS2_Data.DATE.PSS_RX = 0;
	PS2_Data.DATE.PSS_RY = 0;
	PS2_Data.DATE.PSS_LX = 0;
	PS2_Data.DATE.PSS_LY = 0;
}
//判断是否为红灯模式
//返回值;0,红灯模式
//		  其他,其他模式
uint8_t PS2_RedLight(void)
{
	uint8_t Data;
	
	CS_L();
	PS2_Cmd(Comd[0]);  //开始命令
	Data = PS2_Cmd(Comd[1]);  //请求数据
	CS_H();
	PS2_Data.DATE.ID = Data;
	if( Data == 0X73) return 0 ;
	else return 1;
}

//读取手柄数据
void PS2_ReadData(void)
{
	volatile uint8_t byte=0;
	volatile uint16_t ref=0x01;
	uint8_t data;

	CS_L();
	PS2_Cmd(Comd[0]);  //开始命令
	PS2_Data.DATE.ID = PS2_Cmd(Comd[1]);  //请求数据
	for(byte=1;byte<8;byte++)          //开始接受数据
	{
		data = 0;
		for(ref=0x01;ref<0x100;ref<<=1)
		{
			CLK_H();
			delay_us(10);
			CLK_L();
			delay_us(10);
			CLK_H();
		    if(DI())
			{
		       data = ref|data;
			}
		} 
		PS2_Data.byte[byte] = data;
        delay_us(16);
	}
	CS_H();	
}

//对读出来的PS2的数据进行处理      
//只处理了按键部分         
//默认数据是红灯模式  只有一个按键按下时
//按下为0, 未按下为1
uint8_t PS2_DataKey(void)
{
	uint8_t index;
	uint16_t Handkey;

	PS2_ClearData();
	PS2_ReadData();
    if(PS2_Data.DATE.verify == 0x5a)
	{
		Handkey=(PS2_Data.byte[3]<<8)|PS2_Data.byte[2];     //这是16个按键  按下为0, 未按下为1
		for(index=0;index<16;index++)
		{	    
			if((Handkey&(1<<(MASK[index]-1)))==0)
			{
				return index+1;
			}
		}
	}
	return 0;          //没有任何按键按下
}

//得到一个摇杆的模拟量	 范围0~256
//返回1得取成功,0得取失败
uint8_t PS2_AnologData(_e_RockerSelect button,uint8_t *qdata)
{
	PS2_ClearData();
	PS2_ReadData();
	if(PS2_Data.DATE.verify == 0x5a)
	{
		if(button == PSS_RX) *qdata = PS2_Data.DATE.PSS_RX;
		else if(button == PSS_RY) *qdata = PS2_Data.DATE.PSS_RY;
		else if(button == PSS_LX) *qdata = PS2_Data.DATE.PSS_LX;
		else *qdata = PS2_Data.DATE.PSS_LY;
		
		return 1;
	}
	else
	{
		return 0;
	}
}
/******************************************************
Function:    void PS2_Vibration(uint8_t motor1, uint8_t motor2)
Description: 手柄震动函数,
Calls:		 void PS2_Cmd(uint8_t CMD);
Input: motor1:右侧小震动电机 0x00关,其他开
	   motor2:左侧大震动电机 0x40~0xFF 电机开,值越大 震动越大
******************************************************/
void PS2_Vibration(uint8_t motor1, uint8_t motor2)
{
	CS_L();
	delay_us(16);
    PS2_Cmd(0x01);  //开始命令
	PS2_Cmd(0x42);  //请求数据
	PS2_Cmd(0X00);
	PS2_Cmd(motor1);
	PS2_Cmd(motor2);
	PS2_Cmd(0X00);
	PS2_Cmd(0X00);
	PS2_Cmd(0X00);
	PS2_Cmd(0X00);
	CS_H();
	delay_us(16);  
}
//手柄配置初始化
void PS2_SetInit(void)
{
	CS_H();
	CLK_H();
	DO_H();
	PS2_ShortPoll();
	PS2_ShortPoll();
	PS2_ShortPoll();
	PS2_EnterConfing();		//进入配置模式
	PS2_TurnOnAnalogMode();	//“红绿灯”配置模式,并选择是否保存
	PS2_VibrationMode();	//开启震动模式
	PS2_ExitConfing();		//完成并保存配置
	CS_H();
	CLK_H();
	DO_H();
}





bsp_hal_ps2.h

#ifndef __BSP_HAL_PS2_H__
#define __BSP_HAL_PS2_H__

/* 包含头文件 ----------------------------------------------------------------*/
#include "main.h"
/* 宏定义 --------------------------------------------------------------------*/
//These are our button constants
#define PSB_SELECT      1
#define PSB_L3          2
#define PSB_R3          3
#define PSB_START       4
#define PSB_PAD_UP      5
#define PSB_PAD_RIGHT   6
#define PSB_PAD_DOWN    7
#define PSB_PAD_LEFT    8
#define PSB_L2          9
#define PSB_R2          10
#define PSB_L1          11
#define PSB_R1          12
#define PSB_GREEN       13
#define PSB_RED         14
#define PSB_BLUE        15
#define PSB_PINK        16
/* 类型定义 ------------------------------------------------------------------*/
typedef enum
{
	PSS_RX = 0,
	PSS_RY,
	PSS_LX,
	PSS_LY,
}_e_RockerSelect;
typedef union
{
	uint8_t byte;
	struct
	{
		uint8_t SELECT  :1;
		uint8_t L3      :1;
		uint8_t R3      :1;
		uint8_t START   :1;
		uint8_t UP      :1;
		uint8_t RIGHT   :1;
		uint8_t DOWN    :1;
		uint8_t LEFT    :1;
	}bit;
}_u_key1;
typedef union
{
	uint8_t byte;
	struct
	{
		uint8_t L2        :1;
		uint8_t R2        :1;
		uint8_t L1        :1;
		uint8_t R1        :1;
		uint8_t TRI       :1;
		uint8_t CIRCLE    :1;
		uint8_t FORK      :1;
		uint8_t REC       :1;
	}bit;
}_u_key2;
typedef union
{
	uint8_t byte[8];
	struct
	{
		uint8_t ID;
		uint8_t verify;
		_u_key1 key1_U;
		_u_key2 key2_U;
		uint8_t PSS_RX;
		uint8_t PSS_RY;
		uint8_t PSS_LX;
		uint8_t PSS_LY;
	}DATE;
}_u_PS2_Data;

/* 扩展变量 ------------------------------------------------------------------*/
extern _u_PS2_Data PS2_Data;
/* 函数声明 ------------------------------------------------------------------*/
void PS2_SetInit(void);
void PS2_ClearData(void);
void PS2_ReadData(void);
void PS2_Vibration(uint8_t motor1, uint8_t motor2);
uint8_t PS2_AnologData(_e_RockerSelect button,uint8_t *qdata);
uint8_t PS2_DataKey(void);
uint8_t PS2_RedLight(void);

#endif  // __BSP_HAL_PS2_H__


2.主函数

初始化

	PS2_SetInit();
	
	uint8_t a;

主循环文章来源地址https://www.toymoban.com/news/detail-633588.html

		if( !PS2_RedLight()) 
		{//判断手柄是否为红灯模式,是,指示灯LED点亮
			HAL_Delay(50);	 //延时很重要不可去
			for(a=0; a<8; a++)//清除数据缓冲区
				PS2_Data.byte[a] = 0x00;
			PS2_ReadData();
			
//			y_axisbuff = (128 - PS2_Data.DATE.PSS_LY)*2;//最高速度256
//			x_axisbuff = (PS2_Data.DATE.PSS_LX - 128)*2;
//			yawbuff = ( -128+PS2_Data.DATE.PSS_RX)*2;
			
//			if(ABS(y_axisbuff) < 50) y_axisbuff = 0; //限制遥控中间位置的误差
//		    if(ABS(yawbuff) < 50) yawbuff = 0;
//		    if(ABS(x_axisbuff) < 50) x_axisbuff = 0;
		}
		else
		{//判断手柄不是红灯模式,指示灯LED熄灭

			PS2_ClearData();
		}
		HAL_Delay(50);

到了这里,关于stm32 cubemx ps2无线(有线)手柄的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【Mac+CLion+STM32+ST-Link】Mac(apple sillicon)上使用STM32CubeMX和CLion搭建嵌入式开发环境

    Clion 官网安装或者brew安装,我用的是2023.2版本。 stlink server https://www.st.com/en/development-tools/st-link-server.html 不安装的话检测不到stlink。 STM32CubeMX https://www.st.com/en/development-tools/stm32cubeide.html#overviewsecondary=st-get-software 用来快速搭建一个工程。 Arm-toolchain 用来编译和debug的组件。

    2024年02月08日
    浏览(45)
  • STM32 PWM模式与输出比较模式的区别。PWM占空比不生效,在STM32CubeMX中配置PWM的两种模式——蓝桥杯嵌入式

      🎊【蓝桥杯嵌入式】专题正在持续更新中,原理图解析✨,各模块分析✨以及历年真题讲解✨都已更新完毕,欢迎大家前往订阅本专题🎏 🎏【蓝桥杯嵌入式】蓝桥杯第十届省赛真题 🎏【蓝桥杯嵌入式】蓝桥杯第十二届省赛程序真题 🎏【蓝桥杯嵌入式】蓝桥杯第十三届

    2023年04月15日
    浏览(80)
  • 嵌入式毕设分享 stm32与GSM的远程无线智能报警系统(项目开源)

    🔥 这两年开始毕业设计和毕业答辩的要求和难度不断提升,传统的毕设题目缺少创新和亮点,往往达不到毕业答辩的要求,这两年不断有学弟学妹告诉学长自己做的项目系统达不到老师的要求。 为了大家能够顺利以及最少的精力通过毕设,学长分享优质毕业设计项目,今天

    2024年02月20日
    浏览(88)
  • 十三、ESP32PS2摇杆(ADC)

    在上下左右操作PS2摇杆的时候,会检测到数据

    2024年02月14日
    浏览(37)
  • 【嵌入式学习笔记】嵌入式基础9——STM32启动过程

    程序段交叉引用关系(Section Cross References):描述各文件之间函数调用关系 删除映像未使用的程序段(Removing Unused input sections from the image):描述工程中未用到被删除的冗余程序段(函数/数据) 映像符号表(Image Symbol Table):描述各符号(程序段/数据)在存储器中的地址、类

    2024年02月15日
    浏览(87)
  • stm32嵌入式实验考核

    STM32 实验考核题目 1. 利用 STM32 小板实现:控制外接 LED 灯每隔 3 秒钟亮暗变换,同 时在 PC 机上显示 MCU 的计时时间,MCU 的初始时间由 PC 机 方设置。 2. 利用 STM32 小板实现:利用导线外接 GPIO 口模拟 2 个按键输入, 根据输入组合的四种情况,分别控制三色灯四种流水灯效果

    2024年02月03日
    浏览(53)
  • 嵌入式 STM32 通讯协议--MODBUS

    目录 一、自定义通信协议 1、协议介绍 2、网络协议 3、自定义的通信协议  二、MODBUS通信协议 1、概述 2、MODBUS帧结构  协议描述 3、MODBUS数据模型   4、MODBUS事务处理的定义 5、MODBUS功能码  6、功能码定义   7、MODBUS数据链路层 8、MODBUS地址规则  9、MODBUS帧描述 10、MODBUS两种

    2024年02月11日
    浏览(66)
  • STM32的时钟系统(嵌入式学习)

    时钟是指用于计量和同步时间的装置或系统。时钟是嵌入式系统的脉搏,处理器内核在时钟驱动下完成指令执行,状态变换等动作,外设部件在时钟的驱动下完成各种工作,例如:串口数据的发送、AD转换、定时器计数等。因此时钟对于计算机系统是至关重要的,通常时钟系

    2024年02月16日
    浏览(50)
  • 嵌入式——新建STM32工程(标准库)

    目录 一、初识标准库 1.CMSIS标准及库层级关系 2.库文件介绍 (1)Libraries文件夹 ①CMSIS文件夹 ②STM32F10x_Std_Periph_Driver文件夹 ③ 在用库建立一个完整的工程时,还需要添加stm32f10x_it.c、 stm32f10x_conf.h 和 system_stm32f10x.c文件 (2)Project文件夹 (3)Utilities文件夹 3.库各文件之间的关

    2024年01月23日
    浏览(59)
  • STM32串口通信详解(嵌入式学习)

    时钟信号在电子领域中是指用于同步和定时电路操作的周期性信号。它在数字系统和通信系统中起着至关重要的作用,用于协调各个组件之间的数据传输和操作。 时钟信号有以下几个重要的方面: 频率:时钟信号的频率是指单位时间内信号周期的数量。它通常以赫兹(Hz)为

    2024年02月09日
    浏览(69)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包