驱动部分
#include <linux/init.h>
#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/errno.h>
#include <linux/miscdevice.h>
#include <linux/types.h>
#include <linux/io.h>
#include <linux/pwm.h>
#include <linux/fs.h>
/************硬件相关***************/
#include <mach/iomux-mx6q.h>
#define PWM_FREQENCY 1
#define DEV_NAME "mypwm"
#define PWM_DEBUG 1
/*pwm for this my_pwm*/
struct pwm_device *pwm = NULL;
static int my_pwm_open(struct inode *inode, struct file *filp)
{
if(pwm != NULL)
return -EBUSY;
/**
* my_pwm正好挂载在I.MX6的pwm4上,所以这里申请3号(从零开始算)PWM
*/
pwm = pwm_request(3, "my_pwm");
if ( pwm == NULL ) {
printk(KERN_ALERT"my_pwm open error.
");
}
#ifdef PWM_DEBUG
printk(KERN_ALERT"my_pwm opened
");
#endif
return 0;
}
static int my_pwm_release(struct inode *inode, struct file *filp)
{
pwm_disable(pwm); // 关闭pwm
pwm_free(pwm); // 释放pwm
pwm = NULL;
#ifdef PWM_DEBUG
printk(KERN_ALERT"my_pwm closed
");
#endif
return 0;
}
static long my_pwm_ioctl(struct file *filp,
unsigned int cmd, unsigned long arg)
{
if(pwm == NULL)
return -EINVAL;
if(arg < 0)
return -EINVAL;
switch (cmd) {
case PWM_FREQENCY:
if(arg==0)
{
pwm_disable(pwm);
#ifdef PWM_DEBUG
printk(KERN_ALERT"my_pwm off
");
#endif
}
else
{
pwm_config(pwm, 1000000000/arg/2, 1000000000/arg);
pwm_enable(pwm);
#ifdef PWM_DEBUG
printk(KERN_ALERT"my_pwm on
");
#endif
}
break;
default:
break;
}
return 0;
}
static struct file_operations my_pwm_fops = {
.owner = THIS_MODULE,
.unlocked_ioctl = my_pwm_ioctl,
.open = my_pwm_open,
.release = my_pwm_release,
};
static struct miscdevice my_pwm_miscdev =
{
.minor = MISC_DYNAMIC_MINOR,
.name = DEV_NAME,
.fops = &my_pwm_fops,
};
static int __init my_pwm_init(void)
{
//printk(KERN_ALERT"check my_pwm init.
");
misc_register(&my_pwm_miscdev);//misc杂项设备是主设备号为10的驱动设备,自动创建节点
mxc_iomux_v3_setup_pad(MX6Q_PAD_SD1_CMD__PWM4_PWMO);
#ifdef PWM_DEBUG
printk(KERN_ALERT"my_pwm init
");
#endif
return 0;
}
static void __exit my_pwm_exit(void)
{
misc_deregister(&my_pwm_miscdev);
#ifdef PWM_DEBUG
printk(KERN_ALERT"my_pwm removed
");
#endif
}
module_init(my_pwm_init);
module_exit(my_pwm_exit);
MODULE_DESCRIPTION("pwm_my_pwm driver");
MODULE_LICENSE("GPL");
测试程序
#include<stdio.h>
#include<stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <fcntl.h>
#include <poll.h>
#include <signal.h>
#include <unistd.h>
#include <fcntl.h>
#define PWM_SET_FREQ 1
#define PWM_STOP 0
int fd;
int main(int argc, char **argv)//dev state freq
{
char* filename;
long int freq;
filename=argv[1];
if (argc != 4)
{
printf("parm limited!
");
return 0;
}
printf("open %s
",filename);
fd=open(filename,O_RDWR);
if(fd<0)
{
printf("can't open
");
return 0;
}
if(!strcmp("on",argv[2]))
{
printf("PWM ON!
");
freq=atol(argv[3]);
ioctl(fd,PWM_SET_FREQ,freq);
printf("PWM FREQ set:%d Hz
",freq);
}
else if(!strcmp("off",argv[2]))
{
ioctl(fd,PWM_STOP);
printf("PWM stop!
");
}
else
printf("error:param wrong!
");
while(1)
{
sleep(1000);
}
close(fd);
return 0;
}