关于友善之臂出的这款contexA9开发板,目前在网络上的资源较少,特别是内核的,非常之少,鉴于这种情况,我将会写一个系列的驱动来做关于tiny4412这款板子开发的总结。

简单介绍一下:

Tiny4412是一款高性能的四核Cortex-A9核心板,由广州友善之臂设计、生产和发行销售。它采用三星Exynos4412作为主处理器,运行主频可高达1.5GHz,Exynos4412内部集成了Mali-400 MP高性能图形引擎,支持3D图形流畅运行,并可播放1080P大尺寸高清视频。三星旗舰智能手机Galaxy S3即是采用此CPU设计。

我用的是普通版.也就是只有一个串口的.但是核心板是一样的。

好了,介绍完毕,前面的文章我们已经说过了如何编写一个字符设备的驱动程序,这里就不再继续扯字符驱动怎么写,非常简单了,看看就懂了。

我们进入整题,今天,我们需要实现一个LED的驱动程序。在友善之臂的核心板上,有4颗LED灯,如何编写一个驱动程序,让它亮起来,首先我们来看看核心板:

LED灯就位于右上角,第一个和第二个都是电源指示灯,我们不需要管它,我们只管后面那4个LED灯。

如何编写?

1、首先找到板子的原理图,找到对应的引脚。

2、接着打开数据手册,找到对应的寄存器。

3、开始编写LED驱动程序

4、编写makefile

5、插入模块insmod xxx.ko

6、查询主设备号 cat /proc/devices

7、创建设备节点 mknod /dev/xxx c x x

8、执行应用程序app

对应的原理图:

从这里我们可以得出一个结论,LED灯是低电平点亮的,也就是往对应的端口里写0,LED灯就亮了。从最下面一幅图可以知道,我们要找的寄存器是GPIO的GPM4开头的这个寄存器,现在我们进入查数据手册的阶段.

查手册:

我们找到手册的第288页GPIO章节的GPMCON这里:

这是我们要配置端口的模式的IO口,端口有以上的一些状态,在这里我们只考虑输出,也就是只要配置Output那一项就可以了。

我们要配的寄存器有GPM4CON[0],GPM4CON[1],GPM4CON[2],GPM4CON[3],这四位,分别配置成output输出模式.

接下来再看一个GPM4DAT,这个是端口的状态寄存器,对状态寄存器就是写0或者写1,那么LED就被驱动了,我们来看看:

好了,寄存器我们已经找到了,接下来,可以进入写代码的阶段了:

首先编写LED驱动程序:

<span style="font-size:18px;">#include <linux/init.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>
#include <linux/io.h>
#include <asm/uaccess.h>
#include <asm/irq.h>
#include <asm/io.h>
//这个是设备的名称,也就是对应在/dev/test-dev
#define DEV_NAME	"test-dev"
//LED灯IO口的地址,也就是刚刚我们在上面的芯片手册看到的Address
#define GPM4COM     0x110002E0
//定义配置模式的指针变量
volatile unsigned long *led_config = NULL ;
//定义配置状态的指针变量
volatile unsigned long *led_dat = NULL ;
//open方法,对LED灯进行初始化
int led_open(struct inode *inode, struct file *filp)
{
	printk("led_open\n");//上层程序对LED进行Open操作的时候会执行这个函数
	//先对LED的端口进行清0操作
	*led_config &= ~(0xffff);
	//将4个IO口16位都设置为Output输出状态
	*led_config |= (0x1111);
	return 0;
}
//write方法
int led_write(struct file *filp , const char __user *buf , size_t count , loff_t *f_pos)
{
	int val ;
	//注意,这里是在内核中进行操作,我们需要使用copy_from_user这个函数将用户态的内容拷贝到内核态
	copy_from_user(&val , buf , count);
	//以下就是当val是哪个值的时候,led就执行相应的操作,这里不多说
	switch(val)
	{
		case 0 :
		        //对状态寄存器进行赋值,以下雷同
				printk(KERN_EMERG"led1_on\n");
				*led_dat &= ~0x1 ;
				break ;
		case 1 :
				printk(KERN_EMERG"led2_on\n");
				*led_dat &= ~0x2 ;
				break ;
		case 2 :
				printk(KERN_EMERG"led3_on\n");
				*led_dat &= ~0x4 ;
				break ;
		case 3 :
				printk(KERN_EMERG"led4_on\n");
				*led_dat &= ~0x8 ;
				break ;
		case 4 :
				printk(KERN_EMERG"ledall_on\n");
				*led_dat &= ~0xf ;
				break ;
		case 5 :
				printk(KERN_EMERG"ledall_off\n");
				*led_dat |= 0xf ;
				break ;

	}
}
//close方法
int led_close(struct inode *inode, struct file *filp)
{
	printk("led_close\n");
	*led_dat |= 0xf ;  //全灭,因为高电平是灭的,0xf ----> 1111
	return 0;
}
//用ioctl这个方法也可以实现LED的操作的,自己去实现吧
#if 0
long led_ioctl(struct file *filp, unsigned int request, unsigned long arg)
{
	switch(request)
	{
		case 0:
			printk(KERN_EMERG"led1 on\n");
			*led_dat &=~0x1 ;
			break;

		case 1:
			printk(KERN_EMERG"led2 on\n");
			*led_dat &=~0x2 ;
			break;

		case 3:
			printk(KERN_EMERG"led3 on\n");
			*led_dat &=~0xf ;
			break;

		case 4:
			printk(KERN_EMERG"led4 on\n");
			*led_dat &=~0x8 ;
			break ;
		default :
			*led_dat |= 0xf ;
	}
}
#endif
//对方法进行初始化
struct file_operations fops = {
	.owner = THIS_MODULE ,
	.open = led_open,
	.release = led_close,
//	.unlocked_ioctl = led_ioctl,
	.write = led_write,
};
//主设备号
int major ;
//启动函数
static __init int test_init(void)
{
	printk("led_init\n");
	major = register_chrdev(major, DEV_NAME, &fops);
	led_config = (volatile unsigned long *)ioremap(GPM4COM , 16);
	led_dat = led_config + 1 ;
	return 0;
}
//注销函数
static __exit void test_exit(void)
{
	printk("led_exit\n");
	unregister_chrdev(major, DEV_NAME);
	iounmap(led_config);
}

module_init(test_init);
module_exit(test_exit);

MODULE_LICENSE("GPL");
MODULE_AUTHOR("Y.X.YANG");
MODULE_VERSION("2016.1.15");</span>

以上就是led这个设备驱动的编写框架。看不懂的可以去学学linux内核设备驱动再来看就很简单了。其实跟单片机的编程差不了多少的,只不过内核驱动是按照框架来编写的,有所驱动就在这里。

驱动程序编写完了,接下来我们编写上层应用层的程序:

<span style="font-size:18px;">#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int main(int argc, char **argv)
{
	int fd;
	int val = 0 ;
	//打开对应的设备
	fd = open("/dev/test-dev",O_RDWR) ;
	if(-1 == fd)
	{
		printf("open fair!\n");
		return -1 ;
	}
	while(1){
		val = 0 ;
		//写write方法就会调用到驱动程序的led_write
		//最后我们能看到的结果是led灯做流水灯的实现,然后全灭,再周而复始
		write(fd , &val , 4);
		sleep(1);
		val = 1 ;
		write(fd , &val , 4);
		sleep(1);
		val = 2 ;
		write(fd , &val , 4);
		sleep(1);
		val = 3 ;
		write(fd , &val , 4);
		sleep(1);
		val = 5 ;
		write(fd , &val , 4);
		sleep(1);
	}
	return 0;
}</span>

好了,程序已经写完了,我们来看看makefile怎么写.

<span style="font-size:18px;">#将你所写的驱动程序编译成模块形式
obj-m	+= leds.o
#你需要的文件系统
ROOTFS = /disk/A9/filesystem
#你需要的内核
KERNEL = /disk/A9/linux-3.5/
#模块编译
all:
	make -C $(KERNEL) M=`pwd` modules
#模块清除
clean:
	make -C $(KERNEL) M=`pwd` clean
	rm -rf my_led
#模块更新
install:
	make -C $(KERNEL) M=`pwd` modules_install INSTALL_MOD_PATH=$(ROOTFS)
#编译上层app应用程序
my_led:
	arm-linux-gcc my_led.c -o my_led</span>

好了,所有的一切都编写完成,我们来看看接下来的操作:

1、先编译整个工程:执行make命令

2、编译app

3、启动minicom,打开开发板的电源,开发板bootload开始启动

4、开发板内核启动

5、进入文件系统,执行insmod插入模块和显示插入后的模块的操作

6、看看主设备号

从这里可以看到,我们的设备test-dev的主设备号是250

所以我们创建设备节点 : mknod /dev/test-dev c 250 0 ,创建完成后

7、执行app应用程序

在这里,我们可以看到程序开始跑起来了,我们来看看开发板上的led是怎么变化的:

好,写了好多,也整理了不少东西出来了,有点累了,以后还要慢慢写.....今天就到这里...

基于ARM_contexA9 led驱动编程的更多相关文章

  1. 全志A33 linux led驱动编程(附实测参考代码)

    开发平台 * 芯灵思SinlinxA33开发板 淘宝店铺: https://sinlinx.taobao.com/ 嵌入式linux 开发板交流 QQ:641395230 开发平台 * 芯灵思Sinl ...

  2. 嵌入式linux驱动开发之点亮led(驱动编程思想之初体验)

    这节我们就开始开始进行实战啦!这里顺便说一下啊,出来做开发的基础很重要啊,基础不好,迟早是要恶补的.个人深刻觉得像这种嵌入式的开发对C语言和微机接口与原理是非常依赖的,必须要有深厚的基础才能hold的 ...

  3. 驱动编程思想之初体验 --------------- 嵌入式linux驱动开发之点亮LED

    这节我们就开始开始进行实战啦!这里顺便说一下啊,出来做开发的基础很重要啊,基础不好,迟早是要恶补的.个人深刻觉得像这种嵌入式的开发对C语言和微机接口与原理是非常依赖的,必须要有深厚的基础才能hold的 ...

  4. 基于sys文件系统的LED驱动的移植【原创】

    基于RK3188平台LED驱动程序的移植的移植.如有不正确之处,欢迎大家指点. 本文的LED驱动程序不是通过打开设备节点来访问和控制LED的,是通过sys文件系统来控制LED. 板子上有四盏灯以及对应 ...

  5. 基于S3C2440的linux-3.6.6移植——LED驱动【转】

    本文转载自:http://www.voidcn.com/blog/lqxandroid2012/article/p-625005.html 目前的linux版本的许多驱动都是基于设备模型,LED也不例 ...

  6. at91sam9x5 linux 4.1.0下dts驱动编程模型

    测试环境:  CPU: AT91SAM9X35      Linux: Atmel提供的linux-at91-linux4sam_5.3 (Linux-4.1.0) 转载请注明: 凌云物网智科嵌入式实 ...

  7. 简单的led驱动程序设计

    基于ok6410: led驱动程序: vim led.c #include<linux/kernel.h>#include<linux/module.h>#include< ...

  8. Java网络编程和NIO详解9:基于NIO的网络编程框架Netty

    Java网络编程和NIO详解9:基于NIO的网络编程框架Netty 转自https://sylvanassun.github.io/2017/11/30/2017-11-30-netty_introd ...

  9. SPI_FLASH时序描述及驱动编程

    推荐 分享一个朋友的人工智能教程,零基础!通俗易懂!希望你也加入到人工智能的队伍中来! http://www.captainbed.net/strongerhuang Ⅰ.写在前面 前面文章讲述过关于 ...

随机推荐

  1. Hive基本原理及环境搭建

    今天我主要是在折腾这个Hive,早上看了一下书,最开始有点凌乱,后面慢慢地发现,hive其实挺简单的,以我的理解就是和数据库有关的东西,那这样的话对我来说就容易多啦,因为我对sql语法应该是比较熟悉了 ...

  2. Swift基础之音乐播放随机变换着色板

    今天的内容比较简单,我也就不做详细的文字介绍了,直接上代码,希望对大家有所帮助 var audioPlayer = AVAudioPlayer()    //梯度配色    let gradientL ...

  3. Gazebo機器人仿真學習探索筆記(三)機器人模型

    gazebo_models:https://bitbucket.org/osrf/gazebo_models 模型庫下載,可以參考如下命令: ~/Rob_Soft/Gazebo7$ hg clone ...

  4. 剑指Offer——滴滴笔试题+知识点总结

    剑指Offer--滴滴笔试题+知识点总结 情景回顾 时间:2016.9.18 15:00-17:00 地点:山东省网络环境智能计算技术重点实验室 事件:滴滴笔试   总体来说,滴滴笔试内容体量不算多, ...

  5. UNIX网络编程——epoll 系列函数简介、与select、poll 的区别

    前面博客<<UNIX环境高级编程--epoll函数使用详解>>有关于epoll函数的讲解. 一.epoll 系列函数简介 #include <sys/epoll.h> ...

  6. Cocos2D中Action的进阶使用技巧(一)

    大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处. 如果觉得写的不好请多提意见,如果觉得不错请多多支持点赞.谢谢! hopy ;) 大家对Cocos2d中动作的使用大概都很清楚了,其实本身act ...

  7. 利用cocos2d-x实现CandyCrushSaga消除功能

    猴子原创,欢迎转载.转载请注明: 转载自Cocos2D开发网–Cocos2Dev.com,谢谢! 原文地址: http://www.cocos2dev.com/?p=455 昨天没事写了个三消玩玩.已 ...

  8. Java进阶(三十六)深入理解Java的接口和抽象类

    Java进阶(三十六)深入理解Java的接口和抽象类 前言 对于面向对象编程来说,抽象是它的一大特征之一.在Java中,可以通过两种形式来体现OOP的抽象:接口和抽象类.这两者有太多相似的地方,又有太 ...

  9. UNIX环境高级编程——进程间通讯方法整理

    一.无名管道pipe #include <unistd.h> int pipe(int fd [2]) 二.fifo #include <sys/stat.h> int mkf ...

  10. 《java入门第一季》之UDP协议下的网络编程详解

    首先看一下UDP协议的图解: 可以看到,分为发送端和接收端程序. 直接上代码: 发送端程序: import java.io.IOException; import java.net.DatagramP ...