编译aarch64 Linux内核并基于qemu运行
核心流程
首先,本文主要讲述如何编译Linux
内核并在qemu
虚拟机上运行。这里针对的架构是aarch64
。
本文的实验平台是Ubuntu 16.04
。
为了达成目标,我们需要有qemu
、buildroot
和linux
安装包或源码。
首先确保qemu-system-aarch64
命令可用,可以通过在命令行执行qemu-system-aarch64 --version
判断。
下载buildroot
源码,链接见下文平台工具。假设其绝对路径保存在变量BUILD_ROOT_PATH
中。执行以下命令
cd $BUILD_ROOT_PATH
make menuconfig
在弹出的配置界面中,设置Target option ---> Target Architecture
为AArch64 (little endian)
;设置Toolchain ---> Toolchain type
为External toolchain
,这时我们可以看到Toolchain ---> Toolchain
的值为linaro AArch64 xxxx.xx
;设置System configuration ---> Enable root login with password
开启,并设置System configuration ---> Root password
为xxxx
(任意的你喜欢的密码);设置System configuration ---> Run a getty (login prompt) after boot ---> TTY port
的值为ttyAMA0
(这一条非常重要,不然虚拟机可能启动不了);设置Target packages ---> Show packages that are also provided by busybox
开启;设置Target packages ---> Debugging, profiling and benchmark ---> strace
开启;设置Filesystem images ---> cpio the root filesystem
开启。
在配置完成之后,执行
make
注意:这里可能需要配置wget
代理。
生成的rootfs.cpio
在目录$BUILD_ROOT_PATH/output/images
下面。
下载Linux
内核源码,链接见下文平台工具。假设其绝对路径保存在变量LINUX_KERNEL_PATH
中。执行以下命令
cd $LINUX_KERNEL_PATH
ARCH=arm64 CROSS_COMPILE=aarch64-linux-gnu- make defconfig
vim .config
在打开的文件中,找到CONFIG_CMDLINE
这个配置选项,设置其值为console=ttyAMA0
;找到CONFIG_INITRAMFS_SOURCE
这个配置选项,设置其值为$BUILD_ROOT_PATH/output/images/rootfs.cpio
(注意,这里要自己展开变量BUILD_ROOT_PATH
);设置CONFIG_DEBUG_INFO
配置项为y
。
配置结束后,执行以下命令
ARCH=arm64 CROSS_COMPILE=aarch64-linux-gnu- make -j16
执行结束后,我们可以看到生成了一些文件,其中包括$LINUX_KERNEL_PATH/vmlinux
和$LINUX_KERNEL_PATH/arch/arm64/boot/Image
在当前目录创建一个shell
脚本,避免重复输入指令。其文件名为start.sh
,其内容为
qemu-system-aarch64 -machine virt -cpu cortex-a57 \
-machine type=virt -nographic -smp 1 \
-m 2048 \
-kernel ./arch/arm64/boot/Image \
--apend "console=ttyAMA0" \
$1 $2
执行./startup.sh
,这是可以用qemu
启动linux
内核。在进入命令行之前,需要输入buildroot login:
的值,其值为root
,然后需要输入Password:
,这是前文构建rootfs.cpio
的时候,配置项System configuration ---> Root password
的值。然后就可以进入命令行执行以下常用命令了。(注意,需要先cd /
)
如果要退出qemu
,可以先按Ctrl + A
,然后按X
。
为了在主机和qemu
虚拟机之间共享文件,我们可以创建一个目录,其绝对路径为SHARED_FILE_PATH
。然后执行以下命令
cd $BUILD_ROOT_PATH
vim .config
修改BR2_ROOTFS_OVERLAY
配置项的值为$SHARED_FILE_PATH
(注意,自行展开变量)。
保存后执行以下命令重新创建 rootfs.cpio
。
rm $BUILD_ROOT_PATH/output/images/rootfs.*
make
然后需要重新编译内核,即
cd $LINUX_KERNEL_PATH
ARCH=arm64 CROSS_COMPILE=aarch64-linux-gnu- make -j16
依然通过./start.sh
启动虚拟机跑linux
内核,进入命令行后,执行cd /; ls
,我们可以看到$SHARED_FILE_PATH
目录下的文件。
重新执行./start.sh -s -S
进入qemu
的调试状态,然后开一个新的shell
,输入命令
cd $LINUX_KERNEL_PATH
aarch64-linux-gnu-gdb ./vmlinux -ex "target remote :1234"
现在,可以像以往一样使用gdb
进行调试了……
下面介绍如何编译一个linux
内核模块。
cd $SHARED_FILE_PATH
vim hello.c
hello.c
的内容为
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/init.h>
#include <linux/seq_file.h>
#include <linux/proc_fs.h>
#include <linux/sched.h>
int helloinit(void)
{
printk(KERN_DEBUG "hello world !!\n");
return 0;
}
void helloexit(void)
{
printk(KERN_DEBUG "goodbye world !!\n");
}
module_init(helloinit);
module_exit(helloexit);
MODULE_LICENSE("GPL");
然后在当前目录创建Makefile
内容如下
ifneq (${KERNELRELEASE},)
obj-m := hello.o
else
KERNEL_SOURCE := $LINUX_KERNEL_PATH # 注意自行展开变量LINUX_KERNEL_PATH
PWD := $(shell pwd)
default:
${MAKE} -C ${KERNEL_SOURCE} M=${PWD} modules
clean:
${MAKE} -C ${KERNEL_SOURCE} M=${PWD} clean
endif
然后执行交叉编译命令
ARCH=arm64 CROSS_COMPILE=aarch64-linux-gnu- make
OK,重新编译rootfs.cpio
和linux
内核,然后运行qemu
。在进入linux
命令行后,执行以下命令
cd /
insmod hello.ko
dmesg | tail # 可以看到打印信息 hello world !!
rmmod hello.ko
dmesg | tail # 可以看到打印信息 goodbye world !!
相关经验
Linux Kernel
- 使用
qemu
并开启gdb server
功能之后,在gdb
窗口输入b start_kernel
,进入最初的内核初始化函数。
平台工具
参考
- Starting Linux kernel exploration
- Debugging an ARM64 linux kernel using QEMU
- Writing kernel modules
- Debugging Kernel and Modules using GDB
编译aarch64 Linux内核并基于qemu运行的更多相关文章
- 编译x86_64 Linux内核并基于QEMU运行
编译并运行内核镜像 安装包准备 $ sudo apt install git $ sudo apt install build-essential kernel-package fakeroot li ...
- 编译安装linux内核步骤
编译安装linux内核步骤: 一.获取内核源码 源码网址:www.kernel.org 二.解压内核源码 首先以root帐号登录,然后进入/usr/src子目录.如果用户在安装Linux时,安装了内核 ...
- 【转】6.4.6 将驱动编译进Linux内核进行测试
原文网址:http://www.apkbus.com/android-98520-1-1.html 前面几节都是将Linux驱动编译成模块,然后动态装载进行测试.动态装载驱动模块不会随着Android ...
- linux内核系列(一)编译安装Linux内核 2.6.18
1.配置环境 操作系统:CentOS 5.2 下载linux-2.6.18版本的内核,网址:http://www.kernel.org 说明:该编译文档适合2.6.18以上的Linux内核版本,只需所 ...
- 将MPLS编译进linux内核中
系统环境:linux kernel 2.6.35.(此环境是上一篇文章中将ubuntu内核替换后的环境) 编译过程如下: 1)首先需要下载patch文件:linux-kernel-v2.6.35-mp ...
- 将驱动编译进Linux内核
*:first-child { margin-top: 0 !important; } body>*:last-child { margin-bottom: 0 !important; } /* ...
- 如何编译安装Linux内核
操作系统环境 VMware workstation15 Pro ubuntu18.04 LTS 待编译内核5.3.10版本 内核下载地址 kernel.org 环境配置 在正式编译前需要安装部分软件. ...
- 编译最新linux内核(version 4.4.2)
环境:centos6.4 内核版本为2.6.32 目标:编译4.4.2内核,升级到 4.4.2 准备工作: 安装开发库和ncurses库 # yum groupinstall "Develo ...
- armv8(aarch64)linux内核中flush_dcache_all函数详细分析【转】
转自:http://blog.csdn.net/qianlong4526888/article/details/12062809 版权声明:本文为博主原创文章,未经博主允许不得转载. /* * __ ...
随机推荐
- Kali Linux 安装中文输入法
1.设置源 vim /etc/apt/sources.list 添加一行 deb http://mirrors.aliyun.com/kali kali-rolling main non-free c ...
- scala :: , +:, :+ , ::: , ++ 的区别
4 种操作符的区别和联系 :: 该方法被称为cons,意为构造,向队列的头部追加数据,创造新的列表.用法为 x::list,其中x为加入到头部的元素,无论x是列表与否,它都只将成为新生成列表的第一个元 ...
- MySQL8 配置远程连接
引言 MySQL8 默认安装后只有本机能访问,如果需要远程连接 MySQL 将无法访问 查看 root 用户权限 进入 mysql 表 select user,host,plugin from use ...
- Redis 安装和启动
Redis 安装 安装准备: redis 压缩包 官网下载地址:https://redis.io/download 安装步骤: 第一步:安装 gcc 编译器 官网发布的 Redis 压缩包是 C 语言 ...
- GPIO模式用法
浮空,顾名思义就是浮在半空,输入直接与寄存器挂钩: 开漏,输出0的时候 PMOS管导通IO输出Vdd,输出1的时候 NMOS管导通IO输出Vss(Cmos场效应管): 推挽,输出时候电平确定,同样使用 ...
- mysql中的实例、数据库关系简介
MySQL是单进程多线程(而Oracle等是多进程),也就是说MySQL实例在系 统上表现就是一个服务进程,即进程(通过多种方法可以创建多实例,再安装一个端口号不同的mysql,或者通过workben ...
- go输出九九乘法表和金字塔
用go语言实现输出九九乘法表和金字塔: 1.输出九九乘法表: 代码示例: package main import ( "fmt" "time" ) //实现99 ...
- 重新整理 .net core 实践篇————配置系统之盟约[五]
前言 在asp .net core 中我们会看到一个appsettings.json 文件,它就是我们在服务中的各种配置,是至关重要的一部门. 不管是官方自带的服务,还是我们自己编写的服务都是用它来实 ...
- springboot对mybatis的整合
1. 导入依赖 首先新建一个springboot项目,勾选组件时勾选Spring Web.JDBC API.MySQL Driver pom.xml配置文件导入依赖 <!--mybatis-sp ...
- Python+selenium 自动化-启用带插件的chrome浏览器,调用浏览器带插件,浏览器加载配置信息。
Python+selenium 自动化-启用带插件的chrome浏览器,调用浏览器带插件,浏览器加载配置信息. 本文链接:https://blog.csdn.net/qq_38161040/art ...