嵌入式LED驱动程序(韦东山板子)

原创 2016年06月01日 00:03:34
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <asm/uaccess.h>
#include <asm/irq.h>
#include <asm/io.h>
#include <asm/arch/regs-gpio.h>
#include <asm/hardware.h>

static struct class *firstdrv_class;
static struct class_device	*firstdrv_class_dev;

volatile unsigned long *gpfcon = NULL;
volatile unsigned long *gpfdat = NULL;


static int first_drv_open(struct inode *inode, struct file *file)
{
	//printk("first_drv_open\n");
	/* 配置GPF4,5,6为输出 */
	*gpfcon &= ~((0x3<<(4*2)) | (0x3<<(5*2)) | (0x3<<(6*2)));  //先清零
	*gpfcon |= ((0x1<<(4*2)) | (0x1<<(5*2)) | (0x1<<(6*2)));   //再把每位设为01,output
	return 0;
}

static ssize_t first_drv_write(struct file *file, const char __user *buf, size_t count, loff_t * ppos)
{
	int val;

	//printk("first_drv_write\n");

	copy_from_user(&val, buf, count);  //从用户空间向内核空间传递数据,得到应用程序write(fd, &val, 4)中的val值
	                          //	copy_to_user();

	if (val == 1)
	{
		// 点灯
		*gpfdat &= ~((1<<4) | (1<<5) | (1<<6));
	}
	else
	{
		// 灭灯
		*gpfdat |= (1<<4) | (1<<5) | (1<<6);
	}
	
	return 0;
}

static struct file_operations first_drv_fops = {
    .owner  =   THIS_MODULE,    /* 这是一个宏,推向编译模块时自动创建的__this_module变量 */
    .open   =   first_drv_open,     
	.write	=	first_drv_write,	   
};


int major;
static int first_drv_init(void)
{
	major = register_chrdev(0, "first_drv", &first_drv_fops); // 注册, 告诉内核

	firstdrv_class = class_create(THIS_MODULE, "firstdrv");   //创建一个类

	firstdrv_class_dev = class_device_create(firstdrv_class, NULL, MKDEV(major, 0), NULL, "xyz"); /* /dev/xyz 建立设备*/

	gpfcon = (volatile unsigned long *)ioremap(0x56000050, 16);   //从0x56000050开始映射16个字节到gpfcon
	gpfdat = gpfcon + 1;   //+1是按照指针类型的,long占4个字节

	return 0;
}

static void first_drv_exit(void)
{
	unregister_chrdev(major, "first_drv"); // 卸载

	class_device_unregister(firstdrv_class_dev);
	class_destroy(firstdrv_class);      //销毁类
	iounmap(gpfcon);
}

module_init(first_drv_init);
module_exit(first_drv_exit);


MODULE_LICENSE("GPL");



测试程序:

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

/* firstdrvtest on
  * firstdrvtest off
  */
int main(int argc, char **argv)
{
	int fd;
	int val = 1;
	fd = open("/dev/xyz", O_RDWR);   //打开设备,设备在驱动里建立好
	if (fd < 0)
	{
		printf("can't open!\n");
	}
	if (argc != 2)   
	{
		printf("Usage :\n");
		printf("%s <on|off>\n", argv[0]);
		return 0;
	}

	if (strcmp(argv[1], "on") == 0)
	{
		val  = 1;
	}
	else
	{
		val = 0;
	}
	
	write(fd, &val, 4);  //对应驱动程序的first_drv_write(file,buf,count)函数
	return 0;
}


版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

嵌入式驱动编写-点亮LED驱动程序

在开发板上,有三个LED灯.如何通过应用程序点亮这三个灯如何编写驱动程序 操作硬件的时候,我们需要准备开发板的原理图和开发手册,,根据这两个文档来进行配置 ...

跟着韦东山老师学习嵌入式----字符设备驱动程序之poll机制

int poll(struct pollfd *fds,nfds_t nfds, int timeout); 总的来说,Poll机制会判断fds中的文件是否可读,如果可读则会立即返回,返回的值就...

arm驱动程序——点亮led-利用次设备号(韦东山的视频总结及针对linux-2.6.30)

主设备号标识设备对应的驱动程序;而次设备号则用来标识同一类设备中的某个唯一的设备。 利用到的函数及结构在上面几节中都有说出,在此不再叙述。 驱动设备:  #include  #include  #...

arm驱动程序——点亮led(韦东山的视频总结及针对linux-2.6.30)

上面两节只是驱动程序的框架,下面就是点亮led了。 目的:点亮led 步骤:     1.写框架。     2.完善硬件相关操作。 a.看原理图 b.看2440手册 c.写代码 下面是所用到的函数及结...

嵌入式LED驱动程序

#include //配置头文件 #include //内核头文件 #include #include //时钟头文件 #include //用户定义模块初始函数名需引用的头文件 #include ...
  • qlexcel
  • qlexcel
  • 2016年05月31日 23:42
  • 236

arm驱动程序——自动创建设备节点 (韦东山的视频总结及针对linux-2.6.30)

mdev机制根据/sys/下的系统信息来创建设备节点。 自动创建节点的步骤: 1.入口函数中:     a.创建一个类。     b.在类下面创建一个设备。 2.出口函数中:     a.注销类下的设...

韦东山驱动程序代码

  • 2013年09月20日 10:51
  • 7.43MB
  • 下载

第一个嵌入式linux驱动程序——LED

驱动程序如下: #include #include #include #include #include #include #include #include #include ...

arm驱动程序——手动设备节点 (韦东山的视频总结及针对linux-2.6.30.4)

驱动学的越到感觉知识越乱,从头开始理清。 下面是看韦东山老师的视频总结的: 写驱动程序主要是搞清楚驱动的框架, 下面是写驱动的步骤: 1. 写驱动的读写等函数,一般应用程序用到哪些就写哪些 。 2. ...

板子ping不通PC怎么办——韦东山嵌入式Linux学习笔记07

实验环境: 1. JZ2440 2. Win7如果想用 u-boot 的 tftp 下载功能,那么一定要保证板子可以 ping 通 PC.一般来说,家用PC都是通过路由器上网的。针对这种情况,我想...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:嵌入式LED驱动程序(韦东山板子)
举报原因:
原因补充:

(最多只允许输入30个字)