0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

如何在Petalinux创建Linux内核模块?

YCqV_FPGA_EETre 来源:赛灵思中文社区论坛 作者:付汉杰 2021-03-02 11:10 次阅读

创建内核模块

Petalinux可以帮助工程师简化内核模块的创建工作。在petalinux工程目录下,使用命令“ petalinux-create -t modules --name --enable”,能创建Linux内核模块,包括c源代码文件,Makefile,Yocto的bb文件。相关文件放在目录“ project-spec / meta-user / recipes-modules”中,目录结构如下。

hankf@XSZGS4:~/proj/vcu-trd-2020.2-peta-qt/project-spec/meta-user/recipes-modules$ tree

.

└── linux-test-module

├── files

│ ├── COPYING

│ ├── linux-test-module.c

│ └── Makefile

├── linux-test-module.bb

└── README

2 directories, 5 files

Makefile

原始Makefile

原始的Makefile只支持一个源文件,内容如下:

obj-m := linux-module-module.o

MY_CFLAGS += -g -DDEBUG

ccflags-y += ${MY_CFLAGS}

SRC := $(shell pwd)

all:

$(MAKE) -C $(KERNEL_SRC) M=$(SRC)

modules_install:

$(MAKE) -C $(KERNEL_SRC) M=$(SRC) modules_install

clean:

rm -f *.o *~ core .depend .*.cmd *.ko *.mod.c

rm -f Module.markers Module.symvers modules.order

rm -rf .tmp_versions Modules.symvers

原始的Makefile,支持模块在内核源码目录外编译。编译的命令是“$(MAKE) -C $(KERNEL_SRC) M=$(SRC)”。其中-C选项将当前工作目录转移到指定的位置;KERNEL_SRC是Yocto/PetaLinux在文件components/yocto/layers/core/meta/classes/module.bbclass里定义的变量,指定了Linux内核源代码目录,一般是Petalinux工程目录下的子目录“build/tmp/work-shared/zynqmp-generic/kernel-source”。这样编译时先进入内核源代码目录,先执行其中的顶层Makefile。M是内核根目录下的Makefile中使用的变量,让make在构造modules目标之前返回到内核模块源的代码目录。

Linux内核顶层Makefile关于M变量的代码:

# Use make M=dir or set the environment variable KBUILD_EXTMOD to specify the

# directory of external module to build. Setting M= takes precedence.

ifeq ("$(origin M)", "command line")

KBUILD_EXTMOD := $(M)

endif

module.bbclass里定义的变量KERNEL_SRC的代码:

python do_devshell_prepend () {

os.environ['CFLAGS'] = ''

os.environ['CPPFLAGS'] = ''

os.environ['CXXFLAGS'] = ''

os.environ['LDFLAGS'] = ''

os.environ['KERNEL_PATH'] = d.getVar('STAGING_KERNEL_DIR')

os.environ['KERNEL_SRC'] = d.getVar('STAGING_KERNEL_DIR')

os.environ['KERNEL_VERSION'] = d.getVar('KERNEL_VERSION')

os.environ['CC'] = d.getVar('KERNEL_CC')

os.environ['LD'] = d.getVar('KERNEL_LD')

os.environ['AR'] = d.getVar('KERNEL_AR')

os.environ['O'] = d.getVar('STAGING_KERNEL_BUILDDIR')

kbuild_extra_symbols = d.getVar('KBUILD_EXTRA_SYMBOLS')

if kbuild_extra_symbols:

os.environ['KBUILD_EXTRA_SYMBOLS'] = kbuild_extra_symbols

else:

os.environ['KBUILD_EXTRA_SYMBOLS'] = ''

}

语句“obj-m := linux-test-module.o”指示Kbuild将模块编译成Linux内核模块文件linux-test-module.ko。

更多信息可以参考相关文档,比如深入理解 Linux 配置/构建系统是如何工作的,LINUX内核编译学习笔记, Linux 内核模块编译 Makefile, Linux 2.6内核Makefile浅析。

多源文件Makefile

如果有多个源文件,也可以学习Linux内核模块的Makefile写法,使Petalinux的内核模块也支持多个源文件。Linux内核通过$(-objs)包含多个目标文件,使Kbuild编译多个源文件。因此在支持新Makefile里,增加了objs语句。首先通过TEST_FILES列出所有C文件,再将C文件替换成Obj文件赋值给$(TEST_MODULE_NAME)-objs 。

export TEST_MODULE_NAME = linux-test-module

export TEST_FILES = linux-test-module.c linux-test-module-internal.c

$(TEST_MODULE_NAME)-objs = $(patsubst %.c,%.o,$(filter %.c,$(TEST_FILES)))

obj-m += $(TEST_MODULE_NAME).o

MY_CFLAGS += -g -DDEBUG

ccflags-y += ${MY_CFLAGS}

SRC := $(shell pwd)

all:

$(MAKE) -C $(KERNEL_SRC) M=$(SRC)

modules_install:

$(MAKE) -C $(KERNEL_SRC) M=$(SRC) modules_install

clean:

rm -f *.o *~ core .depend .*.cmd *.ko *.mod.c

rm -f Module.markers Module.symvers modules.order

rm -rf .tmp_versions Modules.symvers

内核源码目录内Makefile

有时内核模块也需要在Linux内核源码目录内编译,不需要指定编译命令。KERNELRELEASE是Linux内核源码的顶层Makefile定义的一个变量。如果在内核源码目录内编译,先执行Linux内核源码的顶层Makefile,会定义KERNELRELEASE的具体值。如果加上对KERNELRELEASE的检查,前面的Makefile可以改造成既可以在在内核源码目录内编译,也可以在在内核源码目录外编译。下面的Makefile,增加了检查KERNELRELEASE的语句“ifeq ($(KERNELRELEASE),)”。

export TEST_MODULE_NAME = linux-test-module

export TEST_FILES = linux-test-module.c linux-test-module-internal.c

$(TEST_MODULE_NAME)-objs = $(patsubst %.c,%.o,$(filter %.c,$(TEST_FILES)))

obj-m += $(TEST_MODULE_NAME).o

MY_CFLAGS += -g -DDEBUG

ccflags-y += ${MY_CFLAGS}

ifeq ($(KERNELRELEASE),)

SRC := $(shell pwd)

# KERNEL_src=/proj/hankf/zcu106/rdf0428-zcu106-vcu-trd-2020.2/source/linux-kernel

all:

$(MAKE) -C $(KERNEL_SRC) M=$(SRC)

modules_install:

$(MAKE) -C $(KERNEL_SRC) M=$(SRC) modules_install

clean:

rm -f *.o *~ core .depend .*.cmd *.ko *.mod.c

rm -f Module.markers Module.symvers modules.order

rm -rf .tmp_versions Modules.symvers

endif

实际测试中,不加KERNELRELEASE相关语句,也不影响在Linux内核源码目录内编译。据说旧版本Kbuild才需要的KERNELRELEASE相关语句。


上面的Makefile文件也可以作为一个模板。只需要更改TEST_MODULE_NAME的值“linux-test-module”,和更改TEST_FILES后的文件列表,可以用于编译其它模块。


如果既不在PetaLinux环境里编译,也不再Linux内核源码目录内编译,请再在上面的Makefile文件里定义内核源码目录。

Yocto recipe文件

Yocto recipe文件的扩展名是.bb,它定义recipe需要的文件。

原始bb文件

原始的recipe的bb文件的只含有一个C文件。

SUMMARY = "Recipe for build an external linux-test-module Linux kernel module"

SECTION = "PETALINUX/modules"

LICENSE = "GPLv2"

LIC_FILES_CHKSUM = "file://COPYING;md5=12f884d2ae1ff87c09e5b7ccc2c4ca7e"

inherit module

INHIBIT_PACKAGE_STRIP = "1"

SRC_URI = "file://Makefile

file://linux-test-module.c

file://COPYING

"

S = "${WORKDIR}"

# The inherit of module.bbclass will automatically name module packages with

# "kernel-module-" prefix as required by the oe-core build environment.

新bb文件

由于要支持多个源文件,需要在recipe的bb文件的添加所有C文件。修改其实也很简单。下面的bb文件添加了行“file://linux-test-module-internal.c”以为Yocto/PetaLinux增加文件linux-test-module-internal.c。

SUMMARY = "Recipe for build an external linux-test-module Linux kernel module"

SECTION = "PETALINUX/modules"

LICENSE = "GPLv2"

LIC_FILES_CHKSUM = "file://COPYING;md5=12f884d2ae1ff87c09e5b7ccc2c4ca7e"

inherit module

INHIBIT_PACKAGE_STRIP = "1"

SRC_URI = "file://Makefile

file://linux-test-module.c

file://linux-test-module-internal.c

file://COPYING

"

S = "${WORKDIR}"

# The inherit of module.bbclass will automatically name module packages with

# "kernel-module-" prefix as required by the oe-core build environment.

测试环境UBuntu 18.04
PetaLinux 2020.2

原文标题:【工程师分享】在Petalinux编译多个源文件的Linux内核模块,以及扩展Makefile功能

文章出处:【微信公众号:FPGA开发圈】欢迎添加关注!文章转载请注明出处。

责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 内核
    +关注

    关注

    3

    文章

    1309

    浏览量

    39846
  • Linux
    +关注

    关注

    87

    文章

    10990

    浏览量

    206730

原文标题:【工程师分享】在Petalinux编译多个源文件的Linux内核模块,以及扩展Makefile功能

文章出处:【微信号:FPGA-EETrend,微信公众号:FPGA开发圈】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    微软发布Linux内核Rust模块优化补丁

    在此之前,Linux 内核中要想实现模块初始化,必须先创建一个实例,再将其移至特定内存空间。然而,经过新补丁调整后,各模块可直接在预设定好的
    的头像 发表于 04-02 15:11 214次阅读

    RZ/G2L Linux系统如何添加新的内核模块

    RZ/G2L Linux系统的镜像基于yocto构建,本篇介绍如何添加新的内核模块
    的头像 发表于 01-04 12:19 935次阅读
    RZ/G2L <b class='flag-5'>Linux</b>系统如何添加新的<b class='flag-5'>内核模块</b>

    深入分析Linux kernel安全特性: 内核模块签名

    顾名思义,在开启该功能之后,内核在加载内核模块时,会对内核模块的签名进行检查。
    的头像 发表于 10-18 12:32 1774次阅读

    Linux驱动内核模块参数介绍

    在加载一个.ko模块时,也可以像应用程序那样,通过命令行传入一些参数,这个过程发生在调用模块初始化函数之前。 内核支持的参数类型有: bool 、 invbool (反转值bool类型
    的头像 发表于 09-26 17:01 398次阅读

    内核模块知识:多个源文件编译生成一个内核模块

    多个源文件编译生成一个内核模块 例如,将hello.c和world.c两个c文件编译生成一个叫hello_world.o的目标文件,则在 Makefile 中添加以下两句: obj-m
    的头像 发表于 09-26 16:58 520次阅读

    内核模块依赖什么意思

    通常情况下,一个模块只使用内核导出的符号,自己不导出符号。但是如果一个模块需要提供全局变量或函数给另外的模块使用,那么就需要将这些符号导出。 这在一个驱动调用另一个驱动代码时比较常见,
    的头像 发表于 09-26 16:51 346次阅读

    iTOP-RK3568开发板内核模块实验-设置交叉编译器

    在上一章节我们编写了最简单的 helloworld 驱动程序。有了驱动程序以后,要如何编译并使用驱动呢。编译驱动有俩种方法,分别是将驱动编译成内核和将驱动编译成内核模块。我们先来学习如何将驱动编译成
    发表于 09-07 10:00

    让ChatGPT写一个内核模块

    笔者最近看到这样一篇文章 ,原作者让 ChatGPT 写一个内核模块,要求实现的功能是:每 5 秒向控制台打印一句 "Hello world",并且把编译需要的 Makefile 也一起写出来。
    发表于 08-17 09:30 312次阅读

    linux设备驱动程序开发详解

      Linux 设备驱动属于内核的一部分,Linux 内核的一个模块可以以两种方式被编译和加载:  (1)直接编译进
    发表于 07-19 11:10 3次下载

    AMD Xilinx SoC:定制PetaLinux中的FSBL

    客户为AMD Xilinx SoC创建PetaLinux工程。需要定制PetaLinux中的FSBL
    的头像 发表于 07-10 17:05 671次阅读

    Linux内核模块编程基础知识

    内核(Kernel)在计算机科学中是操作系统最基本的部分,主要负责管理系统资源。它是为众多应用程序提供对计算机硬件的安全访问的一部分软件,这种访问是有限的,并由内核决定一个程序在什么时候对某部分硬件
    发表于 06-08 10:52 1180次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>内核模块</b>编程基础知识

    Linux内核模块管理工具总结

    Linux提供了三种方法查询加载到内核模块,一种方法是直接访问proc虚拟文件系统获取,一种方法则是比较常用的lsmod方法获取,而lsmod的输出其实是基于/proc/modules。另外还有一种就是查看/sys/modul
    发表于 06-07 16:27 577次阅读

    Linux内核模块参数传递与sysfs文件系统

    函数传参的内核传参机制,编写内核程序时只要实现传参接口,用户在加载内核模块时即可传入指定参数,使得内核模块更加灵活。
    发表于 06-07 16:23 1350次阅读

    Linux内核模块间通讯方法

    Linux内核模块间通讯方法非常的多,最便捷的方法莫过于函数或变量符号导出,然后直接调用。默认情况下,模块模块之间、模块
    发表于 06-07 16:23 1690次阅读
    <b class='flag-5'>Linux</b><b class='flag-5'>内核模块</b>间通讯方法

    什么是Linux内核 Linux内核体系结构

    如果Windows已经安装了所有可用的驱动程序,而您只需要打开所需的驱动程序怎么办?这本质上就是内核模块Linux所做的。内核模块,也称为可加载内核模块(LKM),对于保持
    发表于 06-01 09:59 913次阅读
    什么是<b class='flag-5'>Linux</b><b class='flag-5'>内核</b> <b class='flag-5'>Linux</b><b class='flag-5'>内核</b>体系结构