嵌入式linux控制led单个灯亮,嵌入式Linux LED小灯点亮实验

问:怎么写LED驱动程序?

1.搭建一个字符驱动的框架(上一节已经完成)

2.完善硬件的操作

问:驱动里操作硬件寄存器与单片机操作硬件寄存器有什么不一样的地方?

答:单片机操作的寄存器地址是物理地址,驱动里面操作的必须是虚拟地址,因为驱动是内核的一部分,内核里的地址都是虚拟地址。

问:怎么让物理地址转换为虚拟地址?

答:使用ioremap函数,它的功能就是将物理地址映射为虚拟地址,具体怎么映射需要去看linux内存管理等内容。

问:应用程序如果要传数据给内核怎么办?

答:使用copy_from_user函数,同理如果内核要传数据给应用空间的应用程序则使用copy_to_user函数。

详细请参考驱动源码:

[cpp]  view plain

copy

print?

#include

#include

#include

#include

#include

#include

#include

#include

#include //class_create

static struct class *firstdrv_class;

static struct device *firstdrv_device;

volatile unsigned long *gpbcon = NULL;

volatile unsigned long *gpbdat = NULL;

int major;

static int first_drv_open(struct inode * inode, struct file * filp)

{

printk("first_drv_open\n");

/*  LED1,LED2,LED3,LED4对应GPB5、GPB6、GPB7、GPB8

*  配置GPB5,6,7,8为输出

*/

*gpbcon &= ~((0x3<

*gpbcon |= ((0x1<

return 0;

}

static int first_drv_write(struct file * file, const char __user * buffer, size_t count, loff_t * ppos)

{

int val;

printk("first_drv_write\n");

copy_from_user(&val, buffer, count);

if (val == 1)

{

// 点灯

*gpbdat &= ~((1<<5) | (1<<6) | (1<<7) | (1<<8));

}

else

{

// 灭灯

*gpbdat |= (1<<5) | (1<<6) | (1<<7) | (1<<8);

}

return 0;

}

/* File operations struct for character device */

static const struct file_operations first_drv_fops = {

.owner      = THIS_MODULE,

.open       = first_drv_open,

.write      = first_drv_write,

};

/* 驱动入口函数 */

static int first_drv_init(void)

{

/* 主设备号设置为0表示由系统自动分配主设备号 */

major = register_chrdev(0, "first_drv", &first_drv_fops);

/* 创建firstdrv类 */

firstdrv_class = class_create(THIS_MODULE, "firstdrv");

/* 在firstdrv类下创建xxx设备,供应用程序打开设备*/

firstdrv_device = device_create(firstdrv_class, NULL, MKDEV(major, 0), NULL, "xxx");

/* 将物理地址映射为虚拟地址 */

gpbcon = (volatile unsigned long *)ioremap(0x56000010, 16);

gpbdat = gpbcon + 1;

return 0;

}

/* 驱动出口函数 */

static void first_drv_exit(void)

{

unregister_chrdev(major, "first_drv");

device_unregister(firstdrv_device);  //卸载类下的设备

class_destroy(firstdrv_class);      //卸载类

iounmap(gpbcon);                    //解除映射

}

module_init(first_drv_init);  //用于修饰入口函数

module_exit(first_drv_exit);  //用于修饰出口函数

MODULE_AUTHOR("LWJ");

MODULE_DESCRIPTION("Just for Demon");

MODULE_LICENSE("GPL");  //遵循GPL协议

应用测试程序源码:

[cpp]  view plain

copy

print?

#include

#include

#include

#include

#include

#include

/* first_test on

* first_test off

*/

int main(int argc ,char *argv[])

{

int fd;

int val = 0;

fd = open("/dev/xxx",O_RDWR);

if (fd

{

printf("open error\n");

}

if (argc != 2)

{

printf("Usage:\n");

printf("%s \n",argv[0]);

return 0;

}

if(strncmp(argv[1],"on",2) == 0)

{

val = 1;

}

else if (strncmp(argv[1],"off",3) == 0)

{

val = 0;

}

write(fd,&val,4);

return 0;

}

测试步骤:

[cpp]  view plain

copy

print?

[WJ2440]# ls

Qt            driver_test   lib           root          udisk

TQLedtest     etc           linuxrc       sbin          usr

app_test      first_drv.ko  mnt           sddisk        var

bin           first_test    opt           sys           web

dev           home          proc          tmp

[WJ2440]# ls -l /dev/xxx

ls: /dev/xxx: No such file or directory

[WJ2440]# insmod first_drv.ko

[WJ2440]# lsmod

first_drv 2300 0 - Live 0xbf003000

[WJ2440]# ls -l /dev/xxx

crw-rw----    1 root     root      252,   0 Jan  2 00:23 /dev/xxx

[WJ2440]# ./first_test

first_drv_open

Usage:

./first_test

[WJ2440]# ./first_test off

first_drv_open

first_drv_write

[WJ2440]# ./first_test on

first_drv_open

first_drv_write

[WJ2440]#

可发现,当执行下面语句时,开发板上的4个LED同时被熄灭:

[WJ2440]# ./first_test off

可发现,当执行下面语句时,开发板上的4个LED同时被点亮:

[WJ2440]# ./first_test on

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值