首页 > 编程语言 >内核编译 设备驱动 驱动程序

内核编译 设备驱动 驱动程序

时间:2024-10-10 11:49:32浏览次数:12  
标签:驱动程序 demo make dev 编译 内核 include config

内核编译

一、内核编译的步骤

编译步骤: (linux 内核源码的顶层目录下操作 ) 1. 拷贝默认配置到 .config              cp config_mini2440_td35 .config 2. make menuconfig 内核配置       make menuconfig 3. make uImage                             make uImage

二、编译过程遇到的问题 

1、在 make uImage编译时报错:
Can't use 'defined(@array)' (Maybe you should just omit the defined()?) at kernel/timeconst.pl line 373.
/home/linux/linux-2.6.32.2/kernel/Makefile:129: recipe for target 'kernel/timeconst.h' failed
make[1]: *** [kernel/timeconst.h] Error 255
Makefile:878: recipe for target 'kernel' failed
make: *** [kernel] Error 2

修改

 2、修改入口地址

通过tftp下载到开发板后,内核还不能正常启动 

 原因:

加载地址 Load Address:0x30008000

入口地址 Entry Point:要改成0x30008040

因为:uImage前64字节是头文件,要偏移64字节

3、 Image zImage uImage
Image 可以直接使用的内核镜映像 zImage 一段解压程序 + Image 的压缩文件 uImage 64 字节的头信息+ zImage

改成下图 

修改方法

 

再重新编译 

三、向内核中加入新文件

向内核中新加文件(以向 drivers/char 下新加 test.c 文件为例) 1. 在 drivers/char 目录下创建并编辑文件 test.c 2. 修改同层目录下 Makefile ,新增一句 obj-$(CONFIG_TEST) += test.o 3. 修改同层目录下的 Kconfig ,新增一个配置选项 config TEST bool “ this is a test ” default y help hahaha, have big use, donot delelte! 4. make menuconfig 5. make uImage

 四、加入新的目录

1、创建一个新的目录,在该目录下写一个新的Kconfig

Kconfig的格式

2、在该目录的上一级目录的Kconfig里加入

五、补充知识

1、在 Linux 内核编译过程中,cp config_mini2440_td35 .config 命令的作用是将一个预先配置好的内核配置文件 config_mini2440_td35 复制到当前目录下的 .config 文件中。

2、make menuconfig 可视化的配置菜单(内核活地图)配置完成后,可以保存更改,这些更改将反映在当前目录下的 .config 文件中。

3、更改Kconfig的配置后,make menuconfig菜单也会被更改(Kconfig 定义make menuconfig中的选项)

4、

.config ( 存放 make menuconfig 的配置结果 ) CONFIG_BT = y CONFIG_WIFI = n Makefile ( 使用 .config 中的变量 ) obj-$(CONFIG_BT) += bt.o obj-$(CONFIG_WIFI) += wifi.o
b 相对跳转 短跳转 pc += 100 ldr 绝对跳转 长跳转 pc = 200 地址相关代码: 加载地址和链接地址需要保持一致 地址无关代码:

设备驱动

设备驱动分类: 字符设备:数据访问是顺序的 ( 字节流 ) 块设备:数据访问是随机的,一般是存储设备 网络设备:会集成协议, 靠名字维护

 计算机组成:软件(图上)、硬件(图下)

 设备号: 32位 高12位:主设备号,区分不通类型的设备 低20位:次设备号,区分同类的不通设备 

驱动程序开发步骤

1、向内核中加入新文件(具体参考上面) 加入的demo.c文件内容 (驱动程序),加入后配置make menuconfig 编译make uImage
#include <linux/init.h>
#include <linux/kernel.h>
#include <linux/types.h>
#include <linux/fs.h>
#include <linux/cdev.h>
#include <linux/kdev_t.h>
#include <linux/module.h>

#define MAJOR_NUM 255
#define MINOR_NUM 0
#define DEV_NAME "demo"

static int open(struct inode*node,struct file*file)
{
	printk("demo open ...\n");
	return 0;
}

static ssize_t read(struct file*file,char __user*buf,size_t len,loff_t*offset)
{
	printk("demo read...\n");
	return 0;
}

static ssize_t write(struct file*file,const char __user*buf,size_t len,loff_t*offset)
{
	printk("demo write ...\n");
	return 0;
}

static int close(struct inode*node,struct file *file)
{
	printk("demo close...\n");
	return 0;
}

static dev_t dev_num;
static struct  file_operations fops=
{
	.owner =THIS_MODULE,
	.open=open,
	.read=read,
	.write=write,
	.release=close
};
static struct cdev dev;

static int __init demo_init(void)
{
	int ret=0;
	dev_num =MKDEV(MAJOR_NUM,MINOR_NUM);

	ret=cdev_add(&dev,dev_num,1);
	if(ret<0)
	{
		goto err_cdrv_add;
	}

	cdev_init(&dev,&fops);

	ret=register_chrdev_region(dev_num,1,DEV_NAME);
	if(ret<0)
		goto err_register;

	printk("demo_init #######################\n");
	// register
	return 0;

err_cdrv_add:
	cdev_del(&dev);
	printk("demo_init cdev_add failed ...\n");
	return ret;

err_register:
	unregister_chrdev_region(dev_num,1);
	cdev_del(&dev);
	printk("demo_init cdev_add failed ...\n");
	return ret;
}

static void __exit demo_exit(void)
{
	unregister_chrdev_region(dev_num,1);
	cdev_del(&dev);

	printk("demo_exit #######################\n");
}

module_init(demo_init);
module_exit(demo_exit);

2、在 下写应用层的程序,并用 arm-linux-gcc demo_app.c -o demo_app命令编译 应用层的程序demo_app.c的内容
#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>
#include<stdlib.h>

int main(int argc, const char *argv[])
{
	int fd=open("/dev/demo_app",O_RDWR);
	if(-1==fd)
	{
		perror("open");
		return -1;
	}

	char buf[10];
	read(fd,buf,sizeof(buf));
	write(fd,buf,sizeof(buf));

	close(fd);
	
	return 0;
}

3、在开发板下载内核、启动内核

4、在开发板上的Linux系统输入命令:mknod /dev/demo_app c 255 0

手动创建设备节点: mknod /dev/demo c 255 0 /dev/demo 设备节点名称 (open 的设备名 ) c 字符设备 255 主设备号 0 次设备号

 5、最后运行应用层的程序 

标签:驱动程序,demo,make,dev,编译,内核,include,config
From: https://blog.csdn.net/T66656/article/details/142754052

相关文章

  • 手把手教你学PCIE(6.9)--驱动程序开发实例的网络设备驱动程序开发
    目录1.开发环境准备1.1安装开发工具1.2创建项目目录2.驱动程序代码2.1驱动程序头文件2.2驱动程序主文件3.编译驱动程序4.加载和卸载驱动程序5.测试驱动程序6.总结开发一个网络设备驱动程序是一个复杂的任务,涉及到网络协议栈的集成和硬件设备的管理。在......
  • 第二十二章 编译WebRTC框架
    源代码地址:https://webrtc.googlesource.com/src.git下载后放到src目录下gitclone https://chromium.googlesource.com/chromium/src/buildtools下载后放到src目录下gitclone https://chromium.googlesource.com/chromium/src/build需要https://chromium.googlesource.co......
  • Android反编译APK与解析软件包img
    1,Jadx反编译APK下载JadxJADX相当于是apktool+dex2jar+jd-gui的结合体,既能反编译代码也能反编译资源支持的文件有:(apk,dex,jar,class,smali,zip,xapk,aar,arsc,jadx,aab)等。打开之后界面如下,点击打开文件,再选择对应所需要反编译的apk(或者直接拖入apk)等......
  • 快速编译安装nginx服务
    文章目录1软件包下载2安装编译依赖环境3编译安装Nginx4将Nginx做成服务5启动Nginx服务6访问Nginx页面7其他配置1、软件包下载nginx官网下载地址:https://nginx.org/en/download.html2、安装编译依赖环境1、安装编译环境yum -y install gcc gcc-c++2......
  • C++编译并运行后出现Process finished with exit code 139 (interrupted by signal 11
    问题描述:        代码运行意外终止,报错信息为Processfinishedwithexitcode139(interruptedbysignal11:SIGSEGV)CMakeList文件如下:cmake_minimum_required(VERSION3.26)project(SLAM)set(CMAKE_CXX_STANDARD17)set(CMAKE_CXX_STANDARD_REQUIRED......
  • 为什么 React 和 Vue 不采用像 Svelte 那样的编译方式?
    在前端框架的竞争中,Svelte近年来以其极高的性能和轻量级的架构吸引了众多开发者的注意。与React和Vue等传统框架不同,Svelte通过编译时优化实现高效的UI更新,不依赖于虚拟DOM。然而,尽管Svelte的这种方法具有明显的性能优势,React和Vue仍然没有采用类似的编译方式......