linux驱动学习(2)-beep驱动

发布者:心满愿望最新更新时间:2024-08-14 来源: cnblogs关键字:linux  驱动学习 手机看文章 扫描二维码
随时随地手机看文章

//         >0设置down_counter中的数值(鸣叫频率)

//-------------------------------------------------------------

static int xiaoyang_beep_ioctl(struct inode *inode,        struct file *file,        unsigned int cmd,unsigned long arg)

{

unsigned long temp;

if(cmd <= 0)

{

//set as gpb0,output

temp = __raw_readl(S3C2410_GPBCON);        //GPBCON IO Control

temp &= ~3;        //just select the GPBCON[1:0]bits for gpb0

temp |= 1;        //set gpb[0] as output

__raw_writel(temp, S3C2410_GPBCON);

//set gpbdata[0](beep value) as 0

temp = __raw_readl(S3C2410_GPBDAT);        //GPBDAT

temp &= ~1;

__raw_writel(temp, S3C2410_GPBDAT);

}

else

{

//set as TOUT0

temp = __raw_readl(S3C2410_GPBCON);        //GPBCON

temp &= ~3;

temp |= 2;                //using TOUT0(pwm timer)

__raw_writel(temp, S3C2410_GPBCON);

//using pwm timer TOUT0

temp = __raw_readl(S3C2410_TCFG0);        //TCFG0

temp &= ~0xff;        //select Timer0

temp |= 15;                //FCLK/(15+1)/(devided value),prescaler value!

__raw_writel(temp, S3C2410_TCFG0);

temp = __raw_readl(S3C2410_TCFG1);        //TCFG1

temp &= ~0xf;        //select MUX0

temp |= 1;                //1/4 diveder

__raw_writel(temp, S3C2410_TCFG1);

//--------------------------------------------------------------

//        now set TCNTB and TCMPB,generate 1:1 diveded timer

//         TCNTB寄存器设置装载到递减计数器中的初始值,并依次递减

//         TCMPB寄存器则用于装载同递减计数器进行比较的值

//        所以当设置TCMPB = TCNTB/2即表示进行1:1分时

//--------------------------------------------------------------

temp = (50000000/64)/cmd;//set timer count buffer rgister ,50MHZ/(16*4*cmd)

__raw_writel(temp, S3C2410_TCNTB(0));

temp >>= 1;

__raw_writel(temp, S3C2410_TCMPB(0));

//setup timer0

temp = __raw_readl(S3C2410_TCON);        //TCON

temp &= ~0x1f;

temp |= 0xb;//disable deadzone,auto-reload,interval-off,update TCNTB and TCMPB,start timer0

__raw_writel(temp, S3C2410_TCON);

temp &= ~2;//set bit[0] as 0,clear manual update-setting bit

__raw_writel(temp, S3C2410_TCON);        

}

return 0;

}

//--------------------------------------------------------------

//        这个结构是字符设备驱动程序的核心

//        当应用程序操作设备文件时所调用的open、read、write、ioctl等函数,

//        最终会调用这个结构中指定的对应函数

//--------------------------------------------------------------

static struct file_operations xiaoyang_beep_fops = {

.owner        =        THIS_MODULE,

.ioctl        =        xiaoyang_beep_ioctl,

};

static char __initdata banner[] = 'TQ2440 Beep, 2010-xiaoyang yin';

static struct class *beep_class;

//--------------------------------------------------------------

//        执行'insmod xiaoyang_beep.ko'命令时就会调用这个函数

//--------------------------------------------------------------

static int __init xiaoyang_beep_init(void)

{

int ret;

printk(banner);

 

//--------------------------------------------------------------

// 注册字符设备驱动程序

// 参数为主设备号、设备名字、file_operations结构;

// 这样,主设备号就和具体的file_operations结构联系起来了,

// 操作主设备为BEEP_MAJOR的设备文件时,就会调用xiaoyang_beep_fops中的相关成员函数

// BEEP_MAJOR可以设为0,表示由内核自动分配主设备号

//--------------------------------------------------------------

ret = register_chrdev(BEEP_MAJOR, DEVICE_NAME, &xiaoyang_beep_fops);//分配设备号

if (ret < 0) {

printk(DEVICE_NAME ' can't register major numbern');

return ret;

}

//注册一个类,使mdev可以在'/dev/'目录下面建立设备节点

beep_class = class_create(THIS_MODULE, DEVICE_NAME);

if(IS_ERR(beep_class))

{

printk('Err: failed in xiaoyang-Beep class. n');

return -1;

}

//创建一个设备节点,节点名为DEVICE_NAME

device_create(beep_class, NULL, MKDEV(BEEP_MAJOR, 0), NULL, DEVICE_NAME);

printk(DEVICE_NAME ' initializedn');

return 0;

}

//--------------------------------------------------------------

//        执行'rmmod xiaoyang_beep.ko'命令时就会调用这个函数

//--------------------------------------------------------------

static void __exit xiaoyang_beep_exit(void)

{

/* 卸载驱动程序 */

unregister_chrdev(BEEP_MAJOR, DEVICE_NAME);

device_destroy(beep_class, MKDEV(BEEP_MAJOR, 0));                //删掉设备节点

class_destroy(beep_class);                                                                        //注销类

}

//指定驱动程序的初始化函数和卸载函数

module_init(xiaoyang_beep_init);

module_exit(xiaoyang_beep_exit);

/* 描述驱动程序的一些信息,不是必须的 */

MODULE_AUTHOR('software-hit');                                //驱动作者

MODULE_DESCRIPTION('TQ2440 Beep Driver');        //描述信息

MODULE_LICENSE('GPL');                                                //遵循的协议

将其加入内核,配置drivers/char/Kconfig和Makefile将根目录下的config_EmbededSky_W35拷贝到.config然后运行:Make menuconfig进行配置。

可将beep作为模块或者必选(作为内核的一部分)加入内核。则运行

Make oldconfig

Make

生成zImage烧写到开发板后就有/dev/beep设备。

若作为模块作编的话仅仅运行Make SUBDIR=drivers/char/ modules即可编译出EmbededSky_Beep.ko,将其拷贝到开发板insmod加载。

其中在编译过程出现了以下问题:

drivers/char/EmbedSky_beep.c: In function 'xiaoyang_beep_init':

drivers/char/EmbedSky_beep.c:134: error: implicit declaration of function 'class_device_create'

drivers/char/EmbedSky_beep.c: In function 'xiaoyang_beep_exit':

drivers/char/EmbedSky_beep.c:148: error: implicit declaration of function 'class_device_destroy'

make[2]: *** [drivers/char/EmbedSky_beep.o] 错误 1

make[1]: *** [drivers/char] 错误 2

make: *** [drivers] 错误 2

[xiaoyang@localhost linux-2.6.3

解决办法:

这主要是版本间的接口变化引起的。

在Linux2.6中,针对上面的这个问题不同的版本有些修改,使用前要先查看下/.../include/linux/device.h里的函数声明,如我用的是Linux2.6.30,里面就没有class_device_create函数,而直接使用device_create就可以了,而在之前的版本如Linux2.6.15,里面就要用class_device_create函数 。

所以不要使用class_device_create 而改用device_create即可,另外一个接口也是。

编写应用程序:

//----------------------------------------------------------

// xiaoyang@2011.4.21

// beep driver/module test

//----------------------------------------------------------

#include

#include

#include

#include

#include

#include

int main()

{

int fd;

int i = 0;

int m_pwm_val;

 

fd = open('/dev/xiaoyang-beep',O_RDWR);

if(fd < 0){

perror('open device xiaoyang-beep error!');

exit(1);

}

 

for(i = 0; i < 1000; i++){

scanf('%d',&m_pwm_val);

printf('your pwm_val is %dn',m_pwm_val);

ioctl(fd,m_pwm_val,4);//最后这个参数没有什么用,ioctl:linux接口采用统一接口进行设备的操作

if(m_pwm_val == 0){

break;

}

}

 

close(fd);

return 0;        

}

makefie如下:

CROSS_COMPILE=arm-linux-

CC=$(CROSS_COMPILE)gcc

obj-m := hello.o

all:beep

beep:beep.c

$(CC) -o beep beep.c

$(CROSS_COMPILE)strip beep

clean:

rm -rf *.o beep

实验结果截图:

输入数字后beep按照不同的频率发声,太刺耳了。


关键字:linux  驱动学习 引用地址:linux驱动学习(2)-beep驱动

上一篇:linux驱动学习(3)--同步、信号量和自旋锁
下一篇:linux驱动学习(1)-环境及hello world程序

推荐阅读最新更新时间:2024-11-11 10:25

基于ARM Linux的无线音视频对讲系统
  随着数字化和网络化时代的到来,尤其是宽带无线网络的发展,为音视频这样大数据量传输业务在无线网络上的应用提供了契机。同时由于音视频独特的感官特性,使其相关的应用需求也变得越来越迫切。无线多媒体是多媒体和移动通信这两个领域的技术相互融合的产物,成为当今通信领域的一个热点。鉴于Linux内核的开源性,采用其作为操作系统,从而使整个系统具有更好的实时性和稳定性。整个系统以ARM11为核心处理器、采用新一代视频编解码标准H.264进行编译码,并通过无线网络传输音视频。它充分利用S3C6410微处理器内部集成的多媒体编解码器(Multi-Formatvideo Codec,MFC),有效提高了系统的性价比。整个系统为无线多媒体音视频的传输
[单片机]
基于ARM <font color='red'>Linux</font>的无线音视频对讲系统
OK6410A 开发板 (八) 90 linux-5.11 OK6410A linux文件系统导读
文件系统是什么 文件系统是 为了处理信息的,以目录的方式展示这些信息,并供用户增删改查 且用户的接口是文件名 linux 真实文件系统分类 信息存储在内核之中:sysfs,procfs,rootfs,devtmpfs 信息存储在内核之外: 存储在 网络 : nfs cifs 存储在 硬盘 : ext2/3/4 ntfs // vfs 并不是真实文件系统,是个框架 // vfs 存在于内核中 我们如何熟悉 linux下的文件系统 从流程去熟悉 1. 挂载 : 涉及到 super_block 2. 文件操作 : 涉及到 inode 3. search及真实文件系统中层级目录的维护 : 涉
[单片机]
arm驱动linux内核时钟
《 linux内核时钟》涉及内核驱动函数四个,内核结构体一个,分析了内核驱动函数一个;可参考的相关应用程序模板或内核驱动模板一个,可参考的相关应用程序模板或内核驱动一个 一、内核定时器 意义:内核定时器是软件意义上的定时器,最终依赖定时器来实现。时钟中断处理程序会唤起Timer_softirq软中断,运行当前处理器上到期的所有定时器。 二、linux设备驱动编程 linux内核提供一组函数,时钟数据结构;这组函数和数据结构使驱动工程师不用关心具体的软件定时器究竟对应着怎样的内核和硬件行为。 三、数据结构和函数: 1)数据结构 结构体一)Linux在include/linux/timer.h头文件中定义了数据结构timer_list来
[单片机]
Linux已被移植到Atmel的ATmega微控制器
Ubuntu Linux发行版已经被移植到最便宜、最便携的平台中:一个Atmel的ATmega微控制器。 拥有一个20MHz的8位AVR处理器,128KB的闪存和整块16KB的SRAM,ATmega1284P并不是建立微型计算机最合乎逻辑的选择。它是在Arduino原型平台中找到的同样基础的设计:一个微控制器,设计使用它的32引脚接口与如传感器、电机和LED等外部硬件相连接。 德米特里·格林伯格的设计,可以在一个Atmel微控制器上运行完整的GNU/Linux操作系统,尽管速度缓慢 但是,这并没有阻止出生在俄罗斯现居住于加利福尼亚州的程序员德米特里·格林伯格进行向该器件中移植Linux内核和完整的桌面操作系统的挑战。
[单片机]
<font color='red'>Linux</font>已被移植到Atmel的ATmega微控制器
arm linux内核移植裁剪过程概览
微处理器的产生为价格低廉、结构小巧的CPU和外设的连接提供了稳定可靠的硬件架构,这样,限制嵌入式系统发展的瓶颈就突出表现在了软件方面。尽管从八十年代末开始,已经陆续出现了一些嵌入式操作系统(比较著名的有Vxwork、pSOS、Neculeus和Windows CE)。但这些专用操作系统都是商业化产品,其高昂的价格使许多低端产品的小公司望而却步;而且,源代码封闭性也大大限制了开发者的积极性。而Linux的开放性,使得许多人都认为Linux非常适合多数Intemet设备。Linux操作系统可以支持不同的设备和不同的配置。Linux对厂商不偏不倚,而且成本极低,因而很快成为用于各种设备的操作系统。嵌入式linux是大势所趋,其巨大的市场潜
[单片机]
瑞萨电子推出基于RZ/G Linux平台的安全解决方案
瑞萨电子与Linux Foundation 的CIP TM安全工作组合作,旨在将用户所需认证时间缩短达六个月 半导体解决方案供应商瑞萨电子株式会社TSE:6723)今日宣布,将于2019年12月底推出基于RZ/G Linux平台、支持IEC62443-4-2国际安全标准认证的安全解决方案,保护工业控制系统免受网络攻击,并有效缩短用户所需的认证时间。 在当今万物互联的世界,非常强大的安全措施日益重要。制造厂和发电厂等基础设施的工业控制系统一旦受到网络攻击,便可能对人们的生活和经济造成严重损害。因此,制定覆盖工业控制系统所有层面(运营商、系统集成商、设备供应商)和所有参与玩家(如制造厂、化工厂、发电厂、智能电网、交通等与工业、
[物联网]
瑞萨电子推出基于RZ/G <font color='red'>Linux</font>平台的安全解决方案
ARMv7:Linux Kernel引导
1.如果内核镜像是压缩的,需要解压缩,引导的第一步是从解压缩开始:archarmbootcompressedhead.S 2.解压缩之后,内核镜像已经存在于ARM中了,下面开始运行,内核开始运行是从/arch/arm/kernel/head.S开始的,入口代码为: 1 2 3 4 5 6 7 8 .arm __HEAD ENTRY(stext) THUMB( adr r9, BSYM(1f) ) @ Kernel is always entered in ARM. THUMB( bx r9 ) @ If this is a Thumb-2 kernel, THUMB( .
[单片机]
基于ARM—LINUX平台的物联网服务器设计
0 引言 物联网是互联网应用的扩展,是一种新兴的联网技术,其核心是物与物之间的信息通信交流,也是物与人之间的交互控制。物联网技术,主要是利用各种传感器设备,例如:无线技术、射频识别(RFID)技术,各类传感器等技术设备将物理世界中的各种信息,如温度、光强、位置等信息通过网络传输,达到物与物之间、物与人之间的信息交互目的。 要使人们能方便地访问物联网信息,监测和控制各类传感节点和电气设备,一个友好的用户界面是前提。基于B/S架构的解决方案以其客户端通用性成为构建系统的首选。本文将设计开发一个通用的基于ARM处理器平台和Linux嵌入式操作系统的物联网服务器,在硬件平台上,将使用基于XSCALEPXA270处理器的UPTECH
[单片机]
基于ARM—<font color='red'>LINUX</font>平台的物联网服务器设计
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved