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

发布时间:2024-08-14  

//         >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按照不同的频率发声,太刺耳了。


文章来源于:电子工程世界    原文链接
本站所有转载文章系出于传递更多信息之目的,且明确注明来源,不希望被转载的媒体或个人可与我们联系,我们将立即进行删除处理。

我们与500+贴片厂合作,完美满足客户的定制需求。为品牌提供定制化的推广方案、专属产品特色页,多渠道推广,SEM/SEO精准营销以及与公众号的联合推广...详细>>

利用葫芦芯平台的卓越技术服务和新产品推广能力,原厂代理能轻松打入消费物联网(IOT)、信息与通信(ICT)、汽车及新能源汽车、工业自动化及工业物联网、装备及功率电子...详细>>

充分利用其强大的电子元器件采购流量,创新性地为这些物料提供了一个全新的窗口。我们的高效数字营销技术,不仅可以助你轻松识别与连接到需求方,更能够极大地提高“闲置物料”的处理能力,通过葫芦芯平台...详细>>

我们的目标很明确:构建一个全方位的半导体产业生态系统。成为一家全球领先的半导体互联网生态公司。目前,我们已成功打造了智能汽车、智能家居、大健康医疗、机器人和材料等五大生态领域。更为重要的是...详细>>

我们深知加工与定制类服务商的价值和重要性,因此,我们倾力为您提供最顶尖的营销资源。在我们的平台上,您可以直接接触到100万的研发工程师和采购工程师,以及10万的活跃客户群体...详细>>

凭借我们强大的专业流量和尖端的互联网数字营销技术,我们承诺为原厂提供免费的产品资料推广服务。无论是最新的资讯、技术动态还是创新产品,都可以通过我们的平台迅速传达给目标客户...详细>>

我们不止于将线索转化为潜在客户。葫芦芯平台致力于形成业务闭环,从引流、宣传到最终销售,全程跟进,确保每一个potential lead都得到妥善处理,从而大幅提高转化率。不仅如此...详细>>