移植mini2440 2.6.32.2内核LED驱动到3.10.17

#include <linux/miscdevice.h>
#include <linux/delay.h>
#include <asm/irq.h>
#include <mach/regs-gpio.h>
#include <mach/hardware.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/mm.h>
#include <linux/fs.h>
#include <linux/types.h>
#include <linux/delay.h>
#include <linux/moduleparam.h>
#include <linux/slab.h>
#include <linux/errno.h>
#include <linux/ioctl.h>
#include <linux/cdev.h>
#include <linux/string.h>
#include <linux/list.h>
#include <linux/pci.h>
#include <linux/gpio.h>
#include <asm/uaccess.h>
#include <asm/atomic.h>
#include <asm/unistd.h>


#define DEVICE_NAME "leds" //设备名(/dev/leds)
//LED 对应的GPIO 端口列表
static unsigned long led_table[] = 
{
S3C2410_GPB(5), 
S3C2410_GPB(6), 
S3C2410_GPB(7), 
S3C2410_GPB(8), 
};
//LED 对应端口将要输出的状态列表
static unsigned int led_cfg_table[] = {


S3C2410_GPIO_OUTPUT, 
S3C2410_GPIO_OUTPUT, 
S3C2410_GPIO_OUTPUT, 
S3C2410_GPIO_OUTPUT, 
};

/*ioctl 函数的实现
 * 在应用/用户层将通过ioctl 函数向内核传递参数,以控制LED 的输出状态
 */
static int sbc2440_leds_ioctl(
struct inode *inode, 
struct file *file, 
unsigned int cmd, 
unsigned long arg)
{


switch(cmd) {


case 0:
case 1:
if (arg > 4) 
{


return -EINVAL;
}
//根据应用/用户层传递来的参数(取反),通过s3c2410_gpio_setpin
//             函数设置LED 对应的端口寄存器,
//s3c_gpio_setpin(led_table[arg], !cmd);
gpio_set_value(led_table[arg], !cmd);
return 0;
default:
return -EINVAL;
}
}
/*
 * 设备函数操作集,在此只有ioctl 函数,通常还有read,  write,
 open,  close 等,因为本LED 驱动在下面已经
 * 注册为misc 设备,因此也可以不用open/close
 */
static struct file_operations dev_fops = {


.owner = THIS_MODULE, 
.ioctl = sbc2440_leds_ioctl, 
};
/*
 * 把LED 驱动注册为MISC 设备
 */
static struct miscdevice misc = {


.minor = MISC_DYNAMIC_MINOR,  //动态设备号
.name = DEVICE_NAME, 
.fops = &dev_fops, 
};
/*
 * 设备初始化
 */
static int __init dev_init(void)
{


int ret;
int i;
for (i = 0; i < 4; i++) {


//设置LED 对应的端口寄存器为输出(OUTPUT)
s3c2410_gpio_cfgpin(led_table[i],  led_cfg_table[i]);
//设置LED
//              对应的端口寄存器为低电平输出,在模块加载结束后,四个LED
//              应该是全部都是发光
//                      //状态
s3c2410_gpio_setpin(led_table[i],  0);
}
ret = misc_register(&misc); //注册设备
printk (DEVICE_NAME"\tinitialized\n");
//打印初始化信息
return ret;
}
static void __exit dev_exit(void)
{
misc_deregister(&misc);
}
module_init(dev_init);
//模块初始化,仅当使用insmod/podprobe
//命令加载时有用,如果设备不是通过模块方式加载,此处将不会被调用
module_exit(dev_exit);//卸载模块,当该设备通过模块方式加载后,可以通过rmmod
    // 命令卸载,将调用此函数
MODULE_LICENSE("GPL"); //版权信息
MODULE_AUTHOR("RenZhe gzs"); //开发者信息


Makefile文件:


KDIR = /work/src/linux-2.6.32.2/
appgcc = /work/toolschain/arm-2013.05/bin/arm-none-linux-gnueabi-gcc
obj-m := led.o

all:
make -C $(KDIR) M=$(PWD)
sudo cp led.ko /work/nfs/root/root#编译后复制到nfs

clean:
make -C $(KDIR) M=$(PWD) clean

源码已经在2.6内核编译通过,修改Makefile文件后编译

make -C /work/src/linux-3.10.17/ M=/work/drivers/arm_led
make[1]: 正在进入目录 `/work/src/linux-3.10.17'
  LD      /work/drivers/arm_led/built-in.o
  CC [M]  /work/drivers/arm_led/led.o
/work/drivers/arm_led/led.c:80: error: unknown field 'ioctl' specified in initializer
/work/drivers/arm_led/led.c:80: warning: initialization from incompatible pointer type
/work/drivers/arm_led/led.c: In function 'dev_init':
/work/drivers/arm_led/led.c:102: error: implicit declaration of function 's3c2410_gpio_cfgpin'
/work/drivers/arm_led/led.c:107: error: implicit declaration of function 's3c2410_gpio_setpin'
make[2]: *** [/work/drivers/arm_led/led.o] 错误 1
make[1]: *** [_module_/work/drivers/arm_led] 错误 2
make[1]:正在离开目录 `/work/src/linux-3.10.17'
make: *** [all] 错误 2


原因是:在3.10.17内核上file_operations发生了重大的改变:

原先的

  int (*ioctl)(struct inode*, struct file*, unsigned int, unsigned long);

被改为了       

   long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);
        long (*compat_ioctl) (struct file *, unsigned int, unsigned long);

因而在实际驱动中,我们需要将原先的写的ioctl函数头给改成下面的unlocked_ioctl,在file_operations结构体的填充中也是一样。

error: unknown field 'ioctl' specified in initializer问题是由于3.10.17内核之后 去掉了原来的ioctl,添加两个新的成员,所以会出错

long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);

 long (*compat_ioctl) (struct file *, unsigned int, unsigned long);

 所以修改源文件中file_operations内.ioctl 改为 .compat_ioctl 即可


至于警告是因为参数减少了一个,去掉即可


用cgvg查找3.10内核目录发现没有(设置IO口为输出)s3c2410_gpio_cfgpin这个函数,对照/drivers/leds/leds-s3c24xx.c这个驱动文件发现设置输出的函数改为了gpio_direction_output函数原型为gpio_direction_output(unisgned gpio, int value);第一个参数为GPIO,第二个为新gpio_set_value(port_num,0/1)gpio_set_value(port_num,0/1)


s3c2410_gpio_setpin对应函数gpio_set_value(port_num,0/1)设置新状态


以下是转载内容:

在kernel 2.6.36 中已经完全删除了struct file_operations 中的ioctl 函数指针,取而代之的是unlocked_ioctl 。

这个指针函数变了之后最大的影响是参数中少了inode , 不过这个不是问题,因为用户程序中的ioctl对应的系统调用接口没有变化,所以用户程序不需要改变,一切都交给内核处理了,如果想在unlocked_ioctl中获得inode 等信息可以用如下方法:
struct inode *inode = file->f_mapping->host;
struct block_device *bdev = inode->i_bdev;
struct gendisk *disk = bdev->bd_disk;
fmode_t mode = file->f_mode;
struct backing_dev_info *bdi;

这次内核函数的变化引出了一个问题,从ioctl系统调用往后,真正的ioctl调用顺序是什么?为什么compat_ioctl 不被调用?
compat_ioctl被使用在用户空间为32位模式,而内核运行在64位模式时。这时候,需要将64位转成32位。
以下是2.6.36的情况:
  
  
SYSCALL_DEFINE3(ioctl ...) compat_sys_ioctl (是否直接调用compat_ioctl 取决于compat_ioctl 是否存在)
| | |-----> compat_ioctl
|   |
|------>do_vfs_ioctl (下一步的调用取决于file->f_path.dentry->d_inode->i_node)
|            |------>file_ioctl
| |
|-------------------------------->vfs_ioctl
|------->unlock_ioctl
其实compat_ioctl 没有被调用的原因是compat_sys_ioctl 没有被调用,而它没有被调用的原因似乎是压根就没有编译到内核中,因为我没有找到调用这个函数的代码。
unlocked_ioctl 实际上取代了用了很久的ioctl,主要的改进就是不再需要上大内核锁 (调用之前不再先调用lock_kernel()然后再unlock_kernel())
总的来说kernel 开发者正在试图朝移除大内核锁的方向努力,ioctl的移除就是被革命了。相信以后越来越多的内核函数会摆脱大内核锁的依赖,并且大内核锁最终会被移除。

(以上内容为转载)

技术选型 【后端】:Java 【框架】:springboot 【前端】:vue 【JDK版本】:JDK1.8 【服务器】:tomcat7+ 【数据库】:mysql 5.7+ 项目包含前后台完整源码。 项目都经过严格调试,确保可以运行! 具体项目介绍可查看博主文章或私聊获取 助力学习实践,提升编程技能,快来获取这份宝贵的资源吧! 在当今快速发展的信息技术领域,技术选型是决定一个项目成功与否的重要因素之一。基于以下的技术栈,我们为您带来了一份完善且经过实践验证的项目资源,让您在学习和提升编程技能的道路上事半功倍。以下是该项目的技术选型和其组件的详细介绍。 在后端技术方面,我们选择了Java作为编程语言。Java以其稳健性、跨平台性和丰富的库支持,在企业级应用中处于领导地位。项目采用了流行的Spring Boot框架,这个框架以简化Java企业级开发而闻名。Spring Boot提供了简洁的配置方式、内置的嵌入式服务器支持以及强大的生态系统,使开发者能够更高效地构建和部署应用。 前端技术方面,我们使用了Vue.js,这是一个用于构建用户界面的渐进式JavaScript框架。Vue以其易上手、灵活和性能出色而受到开发者的青睐,它的组件化开发思想也有助于提高代码的复用性和可维护性。 项目的编译和运行环境选择了JDK 1.8。尽管Java已经推出了更新的版本,但JDK 1.8依旧是一种成熟且稳定的选择,广泛应用于各类项目中,确保了兼容性和稳定性。 在服务器方面,本项目部署在Tomcat 7+之上。Tomcat是Apache软件基金会下的一个开源Servlet容器,也是应用最为广泛的Java Web服务器之一。其稳定性和可靠的性能表现为Java Web应用提供了坚实的支持。 数据库方面,我们采用了MySQL 5.7+。MySQL是一种高效、可靠且使用广泛的关系型数据库管理系统,5.7版本在性能和功能上都有显著的提升。 值得一提的是,该项目包含了前后台的完整源码,并经过严格调试,确保可以顺利运行。通过项目的学习和实践,您将能更好地掌握从后端到前端的完整开发流程,提升自己的编程技能。欢迎参考博主的详细文章或私信获取更多信息,利用这一宝贵资源来推进您的技术成长之路!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值