Linux之手动编译属于自己的微型内核

对于Linux爱好者来说自己手动定做一个属于自己的Linux系统是一件多么酷的事。

那么接下来我就给大家演示一下怎么制作一个属于自己的Linux吧。

 

为什么要制作属于自己的Linux呢?

我们目前用的Linux系统都是从网上或者官方那里得到的,是属于一种通用版,并不是根据自己的硬件特性定制的,不能最大的发挥Linux系统的优越性。因此这是不能满足我们这些Linux发烧友,制作属于自己的Linux让其发挥最大优越性能吧。

 

准备工作:我们制作的Linux是根据我们的硬件本身量身定制的,因此在制作前我们需要清楚的知道我们的目标主机(我们要安装微型Linux的主机)的硬件信息,制作系统前我们需要在宿主机(制作Linux的主机)上完成对系统的制作

 

 

整理一下心情让我们开始吧!!!

 

 

1、步骤规划

   A、为内核提供CPU+Memory+Disk

   B、提供文件系统

   C、提供bash

   D、提供init, --> bash

   E、提供输入设备

   F、提供TCP/IP协议栈,网络功能

 

一、查看设备硬件信息

[root@jsh ~]# cat /proc/cpuinfo       查看CPU信息
processor: 0
vendor_id: GenuineIntel
cpu family: 6
model: 42
model name: Intel(R) Core(TM) i5-2450M CPU @ 2.50GHz    酷睿i5-2450M的
stepping: 7
cpu MHz: 2494.379
cache size: 3072 KB
physical id: 0
siblings: 2
core id: 0
cpu cores: 2                                             双核心
apicid: 0
initial apicid: 0
fpu: yes
fpu_exception: yes
cpuid level: 13
wp: yes
[root@jsh ~]# cat /proc/meminfo       查看内存信息
[root@jsh ~]# lspci                  查看桥设备信息
[root@jsh ~]# lsusb                  查看usb信息



下载内核源代码:www.kernel.org

我们此次实验使用的是3.13.6版本的内核源码



二、为目标磁盘创建文件系统

 

们制作的Linux系统首先是存放在宿主机的一块硬盘上,制作好后把此硬盘给目标机,因此我们首先需要在宿主机上添加一块硬盘然后对此硬盘创建文件系统。

 

添加硬盘步骤

wKiom1P5y_ijp4vAAAHLKm2-C04228.jpg

wKiom1P5zA3wAMFuAAEouahNQJw217.jpg

wKioL1P5zTmgpScMAAGJgJ4fteA976.jpg

wKioL1P5zV3An4zWAAE_n4flKGU599.jpg

wKiom1P5zFmzO3HTAAE4U-tIXr8933.jpg

注:添加完要重新启动虚拟机才会识别此硬盘

添加完成后接下来就是创建文件系统了。

 

[root@jsh ~]# fdisk /dev/sdb     看好自己添加的是哪一块硬盘,此处我添加的是sbd
Command (m for help): n          创建分区
Command action
   e   extended
   p   primary partition (1-4)
P                              创建主分区
Partition number (1-4): 1           创建第一个主分区
First cylinder (1-2610, default 1): 
Using default value 1
Last cylinder, +cylinders or +size{K,M,G} (1-2610, default 2610): +50M
给第一个主分区分配50M空间用来存放内核
Command (m for help): n           再创建一个分区
Command action
   e   extended
   p   primary partition (1-4)
P                              创建主分区
Partition number (1-4): 2           创建第二个主分区
First cylinder (8-2610, default 8): 
Using default value 8
Last cylinder, +cylinders or +size{K,M,G} (8-2610, default 2610): +512M
给第二个主分区分配512M空间来存放根文件系统
Command (m for help): w         保存并退出
The partition table has been altered!
 
Calling ioctl() to re-read partition table.
Syncing disks.
[root@jsh ~]# mke2fs -t ext4 /dev/sdb{1,2}    格式化sdb1和sdb2,文件系统为ext4


创建两个目录,让这两个分区各自挂载

[root@jsh ~]# mkdir /mnt/{boot,sysroot}      在/mnt下创建boot和sysroot两个目录
[root@jsh ~]# mount /dev/sdb1 /mnt/boot/     把sdb1挂载至/mnt/boot/
[root@jsh ~]# mount /dev/sdb2 /mnt/sysroot/   把sdb2挂载至/mnt/sysroot/

为目标机提供根文件系统

[root@jsh ~]# cd /mnt/sysroot/

注:在此目录下我们有一个脚本,是用来复制程序以及程序所依赖的库文件用的脚本

给大家提供这个脚本的代码

[root@jsh sysroot]# vim bincp.sh
#!/bin/bash
#
target=/mnt/sysroot/
 
[ -d $target ] || mkdir $target
#!/bin/bash
#
target=/mnt/sysroot/
 
[ -d $target ] || mkdir $target
 
preCommand() {
    if which $1 &> /dev/null; then
        commandPath=`which --skip-alias $1`
        return 0
    else
        echo "No such command."
        return 1
    fi
}
 
commandCopy() {
    commandDir=`dirname $1`
    [ -d ${target}${commandDir} ] || mkdir -p ${target}${commandDir}
    [ -f ${target}${commandPath} ] || cp $1 ${target}${commandDir}
}
 
libCopy() {
    for lib in `ldd $1 | egrep -o "/[^[:space:]]+"`; do
        libDir=`dirname $lib`
        [ -d ${target}${libDir} ] || mkdir -p ${target}${libDir}
        [ -f ${target}${lib} ] || cp $lib ${target}${libDir}
    done
}
 
read -p "Plz enter a command: " command
 
until [ "$command" == 'quit' ]; do
 
  if preCommand $command ; then
    commandCopy $commandPath
    libCopy $commandPath
  fi
 
  read -p "Plz enter a command: " command
done

注:若复制此脚本的话那么之前创建的文件路径必须与我们之前创建的路径一样,否则此脚本执行时会报错的。

[root@jsh sysroot]# chmod +x bincp.sh         给此脚本一个执行权限 
[root@jsh sysroot]# bash bincp.sh             执行此脚本
Plz enter a command: bash                   复制bash程序
Plz enter a command: cat                    复制cat命令
Plz enter a command: ls                     复制ls命令
Plz enter a command: ifconfig                复制ifconfig命令
Plz enter a command: quit                   退出
[root@jsh sysroot]#

为目标机提供根下的目录

[root@jsh sysroot]# mkdir -pv etc proc sys dev usr var tmp home root boot sbin




三、为目标磁盘安装grub

[root@jsh ~]# grub-install --root-directory=/mnt /dev/sdb
[root@jsh ~]# ls /mnt/boot/
grub  lost+found                      此时我们就能看到boot目录下有一个grub目录




四、编译内核

 

然后我们给grub提供配置文件就行了,配置文件在内核源码中,因此我们需要解压内核源码

wKiom1P5zXagZrvdAAFFhZJWQHk779.jpg

此内核是我之前下载好的

[root@jsh ~]# tar xf linux-3.13.6.tar.xz -C /usr/src/     我们把它解压到/usr/src/下

wKiom1P5za_QKiJIAADGUtA05-k545.jpg

wKioL1P5zsjC49HZAAFPuEbUcS8422.jpg

wKiom1P5zbKj0tQmAAEJTMiZH8Q210.jpg

[root@jsh linux]# make menuconfig     执行此命令,去选择我们用到的功能

注:执行此步骤会弹出一个图形化界面,如果你的系统不支持图形化操作请自行安装开发环境

wKioL1P5zxfD7CSPAAMoXvT8-Eg173.jpg

wKiom1P5zgPTcOK2AAMVB8OkFvU627.jpg

wKioL1P5zx_S-IpZAAPseBS6UHE798.jpg

wKiom1P5zgrxEksjAAF_Lkvw95Q754.jpg

wKioL1P5z1TBUZJjAAQBlk2cnZM588.jpg

wKiom1P5zkGSdxF2AAMj-vxpQo8934.jpg

wKioL1P5z16iAxvZAAQGpzgadwU268.jpg

wKiom1P5zkrg7XMLAAIGnrZiX-M153.jpg

wKioL1P5z5KSHl1rAAR50pei7ic098.jpg

wKiom1P5zn6xgKdbAAMfLGmORjE542.jpg

wKioL1P5z5nSoPu8AAMf8ELrwnM559.jpg

wKiom1P5zoazB6VOAAOCHt13OWo686.jpg

接下来我们就该选磁盘功能了

wKioL1P50A6wQmLaAAMpJVaVSZw022.jpg

wKiom1P5zvmym5H9AAMZiWc0jU4592.jpg

wKioL1P50BaCuJ7MAAO2WIm4KFQ540.jpg

wKiom1P5zwKw5qlKAAN5XTQRylo259.jpg

wKioL1P50B_B_C1ZAAPRf4SP1vs791.jpg

wKiom1P5zwzx0idsAAL9GcFEwLc338.jpg

wKioL1P50CiDHnriAAM7C1TFi9g770.jpg

wKiom1P5zxay4nptAAO3yB504uA097.jpg

wKioL1P50DOjRO_aAAM07KstRPE417.jpg

wKiom1P5z1DjVEGXAAKxpGr4dKw237.jpg

wKioL1P50Gvi9Vi4AAM19b16ke8466.jpg

wKiom1P5z1aBb_GrAAK7yRzPjrY156.jpg

然后就可以保存退出了

[root@jsh linux]# make bzImage       编译内核

注:编译可能要花费一些时间,我们不防这会儿给提供一下grub配置文件

grub提供配置文件

[root@jsh ~]# cd /mnt/boot/grub
[root@jsh grub]# vim grub.conf
timeout=5
default=0
title ce shi linux (3.13.6)
        root (hd0,0)
        kernel /bzImage ro root=/dev/sda2 init=/bin/bash


然后我们再来看一下已经编译好了

wKiom1P5z6DBD9OOAAEv9iPMseI776.jpg

然后把编译好的内核放在我们之前创建的专门用于存放内核的目录下

[root@jsh linux]# cp arch/x86/boot/bzImage /mnt/boot/
[root@jsh linux]# sync        然后同步一下
[root@jsh boot]# chmod +x /mnt/boot/bzImage    给执行权限

然后把宿主机挂起创建目标主机就可以测试我们制作的系统是否能用了。

创建目标虚拟机

wKiom1P50BKRoX9nAAGly4jsbUk068.jpg

wKioL1P50SvTkXV1AAFMjV7wuVI960.jpg

wKiom1P50BayKS3aAAFUmoVEYpE831.jpg

wKioL1P50S-yL48EAAErbjSVLVA078.jpg

wKioL1P50THSQA8SAAD96Fb5bwQ120.jpg

wKiom1P50BviDuZmAADpHvDPlrs038.jpg

wKiom1P50BziSb0KAAFvIHB4S3o573.jpg

wKioL1P50Tag003xAAGEJBjw_sM598.jpg

wKiom1P50CDji9KHAAEF1PiHLOg385.jpg

wKioL1P50TrAdXhWAAG7sFNIN1E946.jpg

wKiom1P50CTgVL16AAEuZ9HpMnI564.jpg

wKioL1P50T2zz4ksAAFZfjZLmq8471.jpg

wKioL1P50T6BS_RTAAEiYdpkxWg282.jpg

wKiom1P50CiToKTEAAOcVZtRDSA641.jpg

此时我们的系统已经能够正常的启动了,但是由于我们没有做好输入设备的驱动,这里的鼠标键盘是不能用的,接下来我们就开始配置内核的输入设备驱动吧。

 

 

 

首先把目标机关机,然后再次开启宿主机,(注:之前做目标机测试时我们要把宿主机挂起,现在我们再启用宿主机

 

在宿主机下:

[root@jsh ~]# cd /usr/src/linux
[root@jsh linux]# make menuconfig

wKiom1P50ILxviB8AAKL3mW4Tp0038.jpg

wKiom1P50ISxaLarAALttMrp3yI198.jpg

wKioL1P50Z7RDiyJAAL-2e8l6bc488.jpg

wKiom1P50IrTsTk8AAMFJBzG7SY550.jpg

wKiom1P50I6A6lRNAAQKcKM0LAc860.jpg

wKioL1P50arQbBnJAAOkhKjG2d0130.jpg

然后保存退出

[root@jsh linux]# make -j 3    编译
[root@jsh linux]# cp arch/x86/boot/bzImage /mnt/boot/
cp: overwrite `/mnt/boot/bzImage'? y           覆盖原文件
[root@jsh linux]# sync                      同步

这就完成了键盘鼠标的驱动,如果这样启动的话还是有一个问题,我们需要让其启动时自动运行init,因此我们需要在/sbin下编辑文件

[root@jsh linux]# cd /mnt/sysroot/sbin
[root@jsh sbin]# vim init
#!/bin/bash
 
echo -e "Welcom to \033[32mce shi\033[34mlinux."
mount -n -t proc proc /proc
mount -n -t sysfs sysfs /sys
mount -n -o remount,rw /dev/sda2
/bin/bash
[root@jsh sbin]# chmod +x init       给一个执行权限
[root@jsh sbin]# vim /mnt/boot/grub/grub.conf

wKiom1P50Oby6pxSAABjJYA3xJQ803.jpg

然后我们需要移植mount命令

[root@jsh ~]# bash bincp.sh        bincp.sh这个脚本要实现创建
Plz enter a command: mount        移植mount命令
Plz enter a command: umount       移植umount命令
Plz enter a command: quit          退出
[root@jsh ~]#

好了,这样我们就可以启动一下试试了,首先挂起宿主机。然后开启目标机

wKioL1P50izDoZrGAAKdMNEthdA951.jpg

Ok这样就启动了,哈哈只不过字体颜色设置的有点奇怪,嘿嘿。

wKiom1P50TKhdDLGAAIP9PmhO08083.jpg

包括我们移植的命令也可以用哦,是不是有点小激动呢,但是总觉得怪怪的,跟我们的正常的系统还是有很多的不一样。

 

但是这样还有一个问题,我们可以看一下我们的设备文件

wKioL1P50mWD7A0pAAAeGzOA5a8453.jpg

我们的dev下是空的,那怎么识别我们的sda1sda2呢?

注:此时各设备文件是在内核的内存中的

 

 

我们还回到宿主机上

[root@jsh ~]# cd /usr/src/linux
[root@jsh linux]# make menuconfig

wKiom1P50ZiynOB8AAJkQWLmImE051.jpg

wKioL1P50rLgjcOAAALVE3UYXCY403.jpg

wKiom1P50Z2gL3H5AAKWMWkMC7I200.jpg

然后保存退出

[root@jsh linux]# make -j 3
[root@jsh linux]# cp arch/x86/boot/bzImage /mnt/boot/
cp: overwrite `/mnt/boot/bzImage'? y

编辑文件手动挂载

[root@jsh ~]# vim /mnt/sysroot/sbin/init

wKioL1P50uqx0_Y5AACdalR1ulM404.jpg

添加红色方框内容

 

好我们再来试一下,看看上边的设备文件有没有挂载上,首先挂起宿主机,然后再开启目标机。

wKiom1P50fSC6QlZAAJF5fq0XF0861.jpg

OK!成功了,呵呵怎么样是不是小有成就呢。

 

至此,我们的六个步骤中的前五个步骤都已经完成了,接下来就是最后一步了,配置网络功能。

 

在宿主机下

[root@jsh ~]# cd /usr/src/linux
[root@jsh linux]# make menuconfig

wKioL1P501qwPe7mAAJ6oR22RXI520.jpg

wKioL1P5012CNkfqAAJ2DhD-Ro8220.jpg

wKiom1P50kiBBH5-AALtfXGlAYs714.jpg

wKioL1P502LDymMJAANDcXfml7k639.jpg

wKiom1P50k3jFPIRAAKVqCELkrs025.jpg

wKioL1P502jCrQZhAAL8YXiHM1E426.jpg

wKiom1P50lOyQrdHAANeuRZf6XI678.jpg

wKioL1P5027xnG5QAANHQ0gobVg656.jpg

wKiom1P50lnTZQBqAAMhor0kQ3o070.jpg

然后保存退出

[root@jsh linux]# make -j 3
[root@jsh linux]# cp arch/x86/boot/bzImage /mnt/boot/
cp: overwrite `/mnt/boot/bzImage'? y

移植一下网络相关的命令

[root@jsh ~]# bash bincp.sh 
Plz enter a command: route
Plz enter a command: netstat
Plz enter a command: ping
Plz enter a command: quit

然后就能挂起宿主机,启动目标机了

wKioL1P507rDxyJmAAIcbhzVmQg503.jpg


OK是不是很有成就感!!!

你可能感兴趣的:(linux,手动编译内核)