`
jcs130
  • 浏览: 131369 次
  • 性别: Icon_minigender_1
  • 来自: Ottawa
社区版块
存档分类
最新评论

WiFi遥控小车(四):简单直流电机驱动及UDP通信程序

 
阅读更多

之前说过只要能控制高低电平就可以控制电机的正反转,我想通过修改友善官方的LED驱动程序来达到控制GPIO高低电平的目的,但是看了很久都没有怎么看懂,就去网上找看有没有比较好理解的程序,找到一个,原帖地址如下,十分感谢intel版主的无私分享

 

http://www.arm9home.net/read.php?tid-15941.html

原帖作者的小车是通过左右轮子的差速来实现转向的,可我的小车是后轮驱动,靠前轮转向的,所以对源程序做了些修改:

 

#include <linux/miscdevice.h>
#include <linux/delay.h>
#include <asm/irq.h>
//#include <mach/regs-gpio.h>
#include <mach/hardware.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/mm.h>
#include <linux/fs.h>
#include <linux/types.h>
#include <linux/delay.h>
#include <linux/moduleparam.h>
#include <linux/slab.h>
#include <linux/errno.h>
#include <linux/ioctl.h>
#include <linux/cdev.h>
#include <linux/string.h>
#include <linux/list.h>
#include <linux/pci.h>
#include <asm/uaccess.h>
#include <asm/atomic.h>
#include <asm/unistd.h>

#include <mach/map.h>
#include <mach/regs-clock.h>
#include <mach/regs-gpio.h>
#include <mach/gpio.h>

//#include <linux/gpio.h>

#include <plat/gpio-cfg.h>
#include <mach/gpio-bank-e.h>
#include <mach/gpio-bank-k.h>

#define DEVICE_NAME "car"

//控制前后的电机
#define MOTOR_QH_ENA 	S3C64XX_GPM(4)
#define MOTOR_QH_ENB	S3C64XX_GPM(5)
//控制方向的电机
#define MOTOR_FX_ENA	S3C64XX_GPE(3)
#define MOTOR_FX_ENB	S3C64XX_GPE(4)


#define debug 1

static unsigned long motor_table [] = {
	MOTOR_QH_ENA,
	MOTOR_QH_ENB,
	MOTOR_FX_ENA,
	MOTOR_FX_ENB
};

//初始化GPIO口
static int car_motor_init(void)
{
  int ret=0,i;
  	for (i = 0; i < 4; i++) 
	{
	    
		if(gpio_is_valid(motor_table[i])==-EINVAL)
		{
		  printk("ERROR,GPIO used by other devices ! \n");
		  break;
		}
		//上拉GPIO
	  s3c_gpio_setpull(motor_table[i], S3C_GPIO_PULL_UP);
	  //设置为输出
	  s3c_gpio_cfgpin(motor_table[i], S3C_GPIO_OUTPUT);
	  //设置默认值为低电平
	  gpio_set_value(motor_table[i],0);
	 
	}
	
  return ret;
};


static void car_motor_status(void)
{
	printk("MOTOR_QH_ENA=%d\nMOTOR_QH_ENB=%d\nMOTOR_FX_ENA=%d\nMOTOR_FX_ENB=%d\n",
	gpio_get_value(MOTOR_QH_ENA),
	gpio_get_value(MOTOR_QH_ENB),
	gpio_get_value(MOTOR_FX_ENA),
	gpio_get_value(MOTOR_FX_ENB));
	
};

static void car_motor_run(void)
{
  	gpio_set_value(MOTOR_QH_ENA,1);
	gpio_set_value(MOTOR_QH_ENB,0);
};


static void car_motor_stop(void)
{
#if debug
	 car_motor_status();
#endif
  	gpio_set_value(MOTOR_QH_ENA,0);
  	gpio_set_value(MOTOR_QH_ENB,0);
};


static void car_motor_back(void)
{
  	gpio_set_value(MOTOR_QH_ENA,0);
	gpio_set_value(MOTOR_QH_ENB,1);
};

static void car_motor_left(void)
{
  	gpio_set_value(MOTOR_FX_ENA,1);
	gpio_set_value(MOTOR_FX_ENB,0);
};

static void car_motor_right(void)
{
 	gpio_set_value(MOTOR_FX_ENA,0);
	gpio_set_value(MOTOR_FX_ENB,1);
};

static void car_motor_zheng(void)
{
 	gpio_set_value(MOTOR_FX_ENA,0);
	gpio_set_value(MOTOR_FX_ENB,0);
};


static long car_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
{
#if debug
	printk("cmd=%d\n",cmd);
#endif
	long ret=0;
	switch(cmd)
	{
	  case '0':	case 0:
	  //小车停止
	  car_motor_stop();
	  break;
	  
	  case '1':	case 1:
	  //小车前进
	  car_motor_run();
	  break;
	  //小车后退
	  case '2':	case 2:
	  car_motor_back();
	  break;
	  //左转
	  case '3':	case 3:
	  car_motor_left();
	  break;
	  //右转
 	  case '4':	case 4:
	  car_motor_right();  
	  break;
	  //直走
	  case '5':	case 5:	
	  car_motor_zheng();
	  break;
	  //小车后退
	  case '6':	case 6:
	  car_motor_back();
	  break;

		return 0;
	default:
		return -EINVAL;
	}
	return ret;
}

static struct file_operations dev_fops = {
	.owner			= THIS_MODULE,
	.unlocked_ioctl	= car_ioctl,
};

static struct miscdevice misc = {
	.minor = MISC_DYNAMIC_MINOR,
	.name = DEVICE_NAME,
	.fops = &dev_fops,
};

static int __init dev_init(void)
{
	int ret;

	car_motor_init();
//	car_motor_run();
	ret = misc_register(&misc);

	printk (DEVICE_NAME"\tinitialized\n");

	return ret;
}

static void __exit dev_exit(void)
{
	misc_deregister(&misc);
}

module_init(dev_init);
module_exit(dev_exit);
MODULE_LICENSE("GPL");
MODULE_AUTHOR("Lintel.");

 为了测试驱动,根据友善的LED测试程序进行了修改,修改的程序如下:

 

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int main(int argc, char **argv)
{
	int cmd;
	int led_no;
	int fd;

	if (argc != 3 || sscanf(argv[1], "%d", &led_no) != 1 || sscanf(argv[2],"%d", &cmd) != 1 ||
			cmd < 0 || cmd > 9 || led_no < 0 || led_no > 3) {
		fprintf(stderr, "Usage: car led_no 0|1\n");
		exit(1);
	}

	fd = open("/dev/car0", 0);
	if (fd < 0) {
		fd = open("/dev/car", 0);
	}
	if (fd < 0) {
		perror("open device car");
		exit(1);
	}

	printf("CMD=%d\n",cmd);
	ioctl(fd, cmd, led_no);
	close(fd);

	return 0;
}

 这个测试程序有个小问题,当传入的参数为‘0’‘1’时没有问题,但是当参数为‘2’时,怎么也传不进去,所以我把驱动代码的‘6’设置为了和‘2’一样的行为。

 

下面进行连线,按照驱动的GPIO口连接到电机驱动模块上,因为没有买到间距为2mm的接口,先用一根一根的线代替了,小车现在是这个样子:


接下来马上就进行UDP通信部分,我也买了友善的SDWIFI模块,插上就能用,按照文档连上了我的路由器,用ifconfig命令查看开发板ip,开发板和笔记本互相能ping通,接下来就是写程序来监听指定端口上得到的数据并做出相应的反应,这部分不是很难,我是借了本书《嵌入式Linux案例开发指南》按照上面的例程修改的,源码如下:

 

/*
 * main.c
 *
 *  Created on: 2012-10-18
 *      Author: micro
 */
#include<sys/types.h>
#include<sys/socket.h>
#include<string.h>
#include<netinet/in.h>
#include<stdio.h>
#include<stdlib.h>
#include <sys/ioctl.h>
#define MAXLINE 5

#define SERV_PORT 12345

void do_echo(int sockfd, struct sockaddr *pcliaddr, socklen_t clilen) {
	int n;
	socklen_t len;
	char mesg[MAXLINE];
	for (;;) {
		len = clilen;
		printf("Waiting for Data...\n");
		//等待数据
		n = recvfrom(sockfd, mesg, MAXLINE, 0, pcliaddr, &len);
		//将数据返回
		//sendto(sockfd, mesg, n, 0, pcliaddr, len);
		//打印数据
		//printf("%s\n", mesg);
		colcars(mesg[0]);
	}
}
//根据收到的信息执行操作
void colcars(char cmd) {
	if (cmd < '0' || cmd > '10') {
		printf("cmd error\n");
	} else {
		if (cmd == '2') {
//			printf("car back\n");
			car_col('6');
		} else {
			car_col(cmd);
		}
	}
}

void car_col(char cmd) {
	int fd;
	fd = open("/dev/car0", 0);
	if (fd < 0) {
		fd = open("/dev/car", 0);
	}
	if (fd < 0) {
		perror("open device car");
		return;
	}
	ioctl(fd, cmd, 0);
	close(fd);
}

int main(void) {
	int sockfd;
	struct sockaddr_in servaddr, cliaddr;
	//建立socket
	sockfd = socket(AF_INET, SOCK_DGRAM, 0);

	//初始化服务器地址
	bzero(&servaddr, sizeof(servaddr));
	servaddr.sin_family = AF_INET;
	servaddr.sin_addr.s_addr = htonl(INADDR_ANY );
	servaddr.sin_port = htons(SERV_PORT);

	//为socket绑定端口
	if (bind(sockfd, (struct sockaddr *) &servaddr, sizeof(servaddr)) == -1) {
		perror("bind error");
		exit(1);
	}
	//执行服务器程序
	do_echo(sockfd, (struct sockaddr *) &cliaddr, sizeof(cliaddr));
	return 0;

}
         而我之前做过控制端的程序,包括安卓、黑莓上的控制程序现在都可以直接拿过来用,只需要把发送的字符串修改一下就好了,下图是我用台式机向开发版发送指令(在同一个路由器连接下),笔记本的终端只相当于一个通过串口显示的作用。

 


至此,我已经基本把我之前的学习内容总结了一遍,我的小车已经基本实现了通过WiFi来远程控制前后左右运动的目的。但是现在我还不会用PWM信号来调速或者控制舵机,而且现在的程序相对粗糙,控制协议也很简单,还存在着每次重启开发板都要手动连接wifi、手动安装驱动与运行服务器程序等不足,这些都是我接下来所要解决的,另外还有块难啃的骨头——视频实时传输……谢谢大家对于我的关注,我会继续努力~~~

 

 传送门:

WiFi遥控小车(一):基于wicam模块的小车

WiFi遥控小车(二):选择学习&开发平台 

WiFi遥控小车(三):搭建嵌入式Linux开发环境

WiFi遥控小车(四):简单直流电机驱动及UDP通信程序 

 

  • 大小: 91.9 KB
  • 大小: 81.5 KB
分享到:
评论

相关推荐

    WiFi遥控小车51单片机程序

    ### WiFi遥控小车51单片机程序解析 #### 概述 本篇文章将深入解析一个基于AT89C51单片机并通过WiFi模块实现手机APP远程控制的小车项目。该程序通过定义不同的动作(前进、左转、右转、停止)来响应手机端发送的...

    STM32 wifi 遥控小车

    STM32 WiFi遥控小车是一种基于STM32微控制器的智能玩具车,它结合了现代无线通信技术和传感器技术,提供了一种创新的控制方式。在这个项目中,STM32芯片作为核心处理器,负责处理各种输入信号,控制电机,并与WiFi...

    如何确定直流电机驱动的PWM 频率.pdf

    直流电机驱动的PWM频率选择是一个复杂的工程问题,涉及电机、电子调速器、电路设计和系统性能优化等多个方面。为了确定最佳的PWM频率,我们需要了解直流电机的工作原理,PWM信号对电机性能的影响,以及不同类型的...

    weixin_contorl.zip_WiFi遥控_wifi小车_wifi遥控小车_智能小车程序

    【标题】"weixin_contorl.zip"是一个包含WiFi遥控功能的智能小车程序,主要用于实现对wifi小车的无线操控。"WiFi遥控"技术是本程序的核心,它利用无线网络连接,使得用户可以通过WiFi信号远程控制小车的行动。 ...

    如何确定直流电机驱动的PWM频率

    在“圆梦小车强身健体啦”一文中提到的新电机无法在原有的驱动程序下正常工作的问题,可以通过调整PWM频率来解决。具体来说: - 如果新电机属于遥控模型级别的电机,那么原系统中的PWM频率可能偏低,不足以应对较大...

    红外遥控玩具小车驱动程序

    红外遥控玩具小车驱动程序 红外遥控玩具小车驱动程序是一种基于红外编码和解码技术的遥控系统,用于控制玩具小车的运动。该系统由红外编码程序、红外解码程序和小车驱动程序三部分组成。在自动模式下,小车可以完成...

    WIFI遥控小车论文报告.pdf

    1. WIFI技术:文中多次提到“WIFI”,这表明该遥控小车项目与无线网络通信技术密切相关。具体技术应用可能包括利用IEEE 802.11标准的无线局域网技术,以实现小车与控制端(如智能手机、平板电脑等)的远程无线控制和...

    智能小车设计电机控制技术.ppt

    本节课程主要讲解智能小车设计电机控制技术,涵盖了电机控制技术的基本概念、直流电机驱动、步进电机驱动、电池相关知识录目等内容。 一、电机控制技术 电机控制技术是智能小车设计的核心技术之一,涉及到电机的...

    Arduino开发遥控小车(三)基于nRF24L01无线模块实现舵机转向和直流电机调速.docx

    在本篇“Arduino开发遥控小车(三)基于nRF24L01无线模块实现舵机转向和直流电机调速”中,我们将深入探讨如何使用Arduino和nRF24L01无线模块来控制遥控小车的舵机转向和直流电机速度。nRF24L01是一款低功耗、高速率...

    STC89C52RC单片机智能小车红外遥控和测速实验程序源代码.rar

    该程序源代码用于STC89C52RC...3、智能小车电机驱动芯片是:L293D; 4、智能小车电机为:TT直流减速电机; 5、需要用到红外遥控器; 6、需要用到测速模块。 该程序源代码在本人STC89C52RC单片机智能小车上亲测可用。

    无线视频遥控的WIFI智能小车

    标题中的“无线视频遥控的WIFI智能小车”是一个基于Arduino技术的项目,它通过WiFi模块实现了远程视频控制的功能。这种技术在物联网(IoT)领域中应用广泛,特别是对于远程监控、自动化控制以及机器人技术等领域有着极...

    WiFi遥控小车代码.docx

    该文档是关于如何使用Arduino和ESP8266 WiFi模块(如ESP-01或NodeMCU)通过WiFi控制一个配备了L298N电机驱动模块的智能小车的代码示例。L298N模块常用于控制直流电机,可以提供足够的电流来驱动小车的马达。 首先,...

    Arduino智能小车红外遥控实验程序源代码.rar

    该程序源代码用于Arduino智能小车红外遥控...3、智能小车电机驱动芯片是:L293D; 4、智能小车电机为:TT直流减速电机; 5、需要用到红外遥控器和红外遥控信号接收管; 该程序源代码在本人Arduino智能小车上亲测可用。

    红外遥控小车设计

    - **电机驱动**: 使用适当的电机驱动电路来控制直流电机或步进电机,以实现小车的前进、后退和转向。 - **软件编程**: 编写相应的控制程序,让单片机能够根据接收到的信号执行相应的指令,如启动、停止或改变速度...

    手机端通过wifi遥控小车stm32程序和android程序,实现前进和后退的功能。

    3. **电机驱动电路**:STM32通过GPIO口输出PWM信号来控制电机驱动器,进而驱动小车的直流电机。电机驱动器的选择需考虑到电机的工作电压、电流需求以及效率。 在Android端,我们需要开发一个应用程序(APP),实现...

    基于51单片机WiFi智能小车制作.pdf

    基于51单片机WiFi智能小车制作是一种智能小车系统,通过使用51单片机、WiFi模块、路由器和电机驱动模块等组件,实现小车的远程控制和自动化驾驶。下面是该系统的详细知识点: 一、基本原理 * 51单片机WiFi智能小车...

    Arduino智能小车红外遥控和测速实验程序源代码.rar

    3、智能小车电机驱动芯片是:L293D; 4、智能小车电机为:TT直流减速电机; 5、需要用到测速模块; 6、需要用到1602液晶模块; 7、需要用到红外遥控器和红外遥控信号接收管。 该程序源代码在本人Arduino智能小车上亲...

    无线通信和遥控小车的C程序及应用

    在无线通信和遥控小车的C程序及应用中,我们主要探讨的是如何通过无线技术来控制小型车辆,这种技术在物联网(IoT)和自动化领域有着广泛的应用。C语言是编程的基础,也是嵌入式系统中常用的编程语言,因此在这个项目...

    红外遥控小车

    红外线接收器用于接收红外线信号,直流电机用于驱动小车的运动,PWM调速模块用于控制小车的速度。 程序中的setup函数用于初始化硬件组件,包括设置红外线接收器、直流电机和PWM调速模块。loop函数用于接收红外线...

Global site tag (gtag.js) - Google Analytics