所属类别:Linux
文章作者:kjpioo
特别推荐:免费发布信息 承包关键词~~抢爆了!HOT!
之前可以轻松编译uImage了。s3c_buttons还不是独立的模块。经过一个下午,重新编译内核。还是没有成功将下面的s3c_buttons.ko驱动从内核中分离出来(没有修改根目录下的Makefile的标志(也就是# Decide whether to build built-in, modular, or both.# Normally, just do built-in.KBUILD_MODULES :=KBUILD_BUILTIN := 1)和s3c_buttons所在文件夹的Makefile文件中的 CONFIG_S3C_BUTTONS += sub-m并且已经通过make modules得到了s3c_buttons.ko模块,但是原来的s3c_buttons预编译进内核的uImage内核)。[root@avantech /]# insmod /lib/s3c_buttons.kokobject_add failed for s3c-buttons with -EEXIST, don't try to register things with the same name in the same directory.insmod: cannot insert '/lib/s3c_buttons.ko': File exists (-1): File exists我在http://hi.csdn.net/iamcow/profile/notebook/1.html发现的读按键的code的程序。源程序是这样的:------------------------------------------------------#include #include #include #include #include typedef struct input_event Myinputevent;#define DEVICENAME "/dev/input/event0"int main(int argc,char **argv){ int i,n; int fd; unsigned char data[8]; Myinputevent event; fd=open(DEVICENAME,O_RDONLY,0); if(fd<0){ perror("Can't open device...\n"); exit(1);} while(1){ printf("Waitting event... \n\n\n"); n=read(fd,&event,sizeof(Myinputevent)); if(n!=16){ perror("Read data error!"); exit(1); } if(event.type==EV_KEY) { if((event.code!=KEY_ESC)&&(event.code!=KEY_UNKNOWN)) printf("Keyboard key ,the code is :%d \n ",event.code); if(event.value) printf(" Press down.\n" ); else printf(" Press up .\n"); } printf("Exit , please press: Ctrl+C \n"); } return 0;}------------------------------------------------------作者说:“你用下面的程序试试,能不能读出正确的键值,我读出来的键值正好是原值”,他是在回答一个问题:“我嵌入式设备上的Linux系统下使用usb键盘直接输入没有字符显示cat /dev/input/event0时,按键,会输出一堆乱码这能说明的usb键盘已经驱动上了吗?请问我如何能获得正确的输入?”原文网址在:http://topic.csdn.net/u/20080803/21/d91619a7-ac2a-4c88-94d6-31e1f0f3a30c.html将这个程序用arm-linux-gcc-3.4.1交叉编译,(名字叫input_evt_test)再下载到arm上,运行。------------------------------------------------------[root@avantech /]# cp /mnt/test/tmp/delete/input_evt_test ./[1] + Done(127) AT S7=45 S0=0 L1 V1 X4[root@avantech /]# lsarm input_evt_test opt usrbin lib proc vardev linuxrc sbinetc lost+found syshome mnt tmp[root@avantech /]# ./input_evt_testWaitting event...Keyboard key ,the code is :3 Press down.Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :3 Press up .Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :5 Press down.Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :5 Press up .Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :2 Press down.Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :2 Press up .Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :4 Press down.Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :4 Press up .Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :4 Press down.Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :4 Press up .Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :3 Press down.Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :3 Press up .Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :5 Press down.Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :5 Press up .Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :5 Press down.Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :5 Press up .Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :2 Press down.Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :2 Press up .Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :4 Press down.Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...Keyboard key ,the code is :4 Press up .Exit , please press: Ctrl+CWaitting event...Exit , please press: Ctrl+CWaitting event...[这里在linux minicom终端上用pc的键盘发出 Ctrl+C信息][root@avantech /]#------------------------------------------------------可见得到了按键扫描码。不知道为什么每按下或抬起都会多输出两行:Exit , please press: Ctrl+CWaitting event...参考资料:///////////////////////////////////////////////////////////////////////////////原文出自:http://www.linuxforum.net作者:jkl==========================================Makefile 初探==========================================Linux的内核配置文件有两个,一个是隐含的.config文件,嵌入到主Makefile中;另一个是 include/linux/autoconf.h,嵌入到各个c源文件中,它们由make config、make menuconfig、make xconfig这些过程创建。几乎所有的源文件都会通过linux/config.h而嵌入autoconf.h,如果按照通常方法建立文件依赖关系 (.depend),只要更新过autoconf.h,就会造成所有源代码的重新编绎。为了优化make过程,减少不必要的重新编绎,Linux开发了专用的mkdep工具,用它来取代gcc来生成.depend文件。mkdep在处 理源文件时,忽略linux/config.h这样的头文件,识别源文件宏指令中具有"CONFIG_"特征的行。例如,如果有"#ifdef CONFIG_SMP"这样的行,它就会在.depend文件中输出$(wildcard /usr/src/linux/include/config/smp.h)。include/config/下的文件是另一个工具split-include从autoconf.h中生成,它利用autoconf.h中的 CONFIG_标记,生成与mkdep相对应的文件。例如,如果autoconf.h中有"#undef CONFIG_SMP"这一行,它就生成include/config/smp.h文件,内容为"#undef CONFIG_SMP"。这些文件名只在.depend文件中出现,内核源文件是不会嵌入它们的。每配置一次内核,运行split-include一次。 split-include会检查旧的子文件的内容,确定是不是要更新它们。这样,不管autoconf.h修改日期如何,只要其配置不变,make就不 会重新编绎内核。如果系统的编绎选项发生了变化,Linux也能进行增量编绎。为了做到这一点,make每编绎一个源文件时生成一个flags文件。例如编绎 sched.c时,会在相同的目录下生成隐含的.sched.o.flags文件。它是Makefile的一个片断,当make进入某个子目录编绎时,会 搜索其中的flags文件,将它们嵌入到Makefile中。这些flags代码测试当前的编绎选项与原来的是不是相同,如果相同,就将自已对应的目标文 件加入FILES_FLAGS_UP_TO_DATE列表,然后,系统从编绎对象表中删除它们,得到FILES_FLAGS_CHANGED列表,最后, 将它们设为目标进行更新。下一步准备逐步深入的剖析Makefile代码。==========================================Makefile解读之二: sub-make==========================================Linux各级内核源代码的子目录下都有Makefile,大多数Makefile要嵌入主目录下的Rule.make,Rule.make将识别各个 Makefile中所定义的一些变量。变量obj-y表示需要编绎到内核中的目标文件名集合,定义O_TARGET表示将obj-y连接为一个 O_TARGET名称的目标文件,定义L_TARGET表示将obj-y合并为一个L_TARGET名称的库文件。同样obj-m表示需要编绎成模块的目 标文件名集合。如果还需进行子目录make,则需要定义subdir-y和subdir-m。在Makefile中,用"obj-$ (CONFIG_BINFMT_ELF) += binfmt_elf.o"和"subdir-$(CONFIG_EXT2_FS) += ext2"这种形式自动为obj-y、obj-m、subdir-y、subdir-m添加文件名。有时,情况没有这么单纯,还需要使用条件语句个别对 待。Makefile中还有其它一些变量,如mod-subdirs定义了subdir-m以外的所有模块子目录。Rules.make是如何使make进入子目录的呢? 先来看subdir-y是如何处理的,在Rules.make中,先对subdir-y中的每一个文件名加上前缀"_subdir_"再进行排序生成 subdir-list集合,再以它作为目标集,对其中每一个目标产生一个子make,同时将目标名的前缀去掉得到子目录名,作为子make的起始目录参 数。subdir-m与subdir-y类似,但情况稍微复杂一些。由于subdir-y中可能有模块定义,因此利用mod-subdirs变量将 subdir-y中模块目录提取出来,再与subdir-m合成一个大的MOD_SUB_DIRS集合。subdir-m的目标所用的前缀是 "_modsubdir_"。一点说明,子目录中的Makefile与Rules.make都没有嵌入.config文件,它是通过主Makefile向下传递 MAKEFILES变量完成的。MAKEFILES是make自已识别的一个变量,在执行新的Makefile之前,make会首先加载 MAKEFILES所指的文件。在主Makefile中它即指向.config。==========================================Makefile解读之三: 模块的版本化处理==========================================模块的版本化是内核与模块接口之间进行严格类型匹配的一种方法。当内核配置了CONFIG_MODVERSIONS之后,make dep操作会在include/linux/modules/目录下为各级Makefile中export-objs变量所对应的源文件生成扩展名为. ver的文件。例如对于kernel/ksyms.c,make用以下命令生成对应的ksyms.ver:gcc -E -D__KERNEL__ -D__GENKSYMS__ ksyms.c /sbin/genksyms -k 2.4.1 >; ksyms.ver-D__GENKSYMS__的作用是使ksyms.c中的EXPORT_SYMBOL宏不进行扩展。genksyms命令识别EXPORT_SYMBOL()中的函数名和对应的原型,再根据其原型计算出该函数的版本号。例如ksyms.c中有一行:EXPORT_SYMBOL(kmalloc);kmalloc原型是:void *kmalloc(size_t, int);genksyms程序对应的输出为:#define __ver_kmalloc 93d4cfe6#define kmalloc _set_ver(kmalloc)在内核符号表和模块中,kmalloc将变成kmalloc_R93d4cfe6。在生成完所有的.ver文件后,make将重建include/linux/modversions.h文件,它包含一系列#include指令行 嵌入各个.ver文件。在编绎内核本身export-objs中的文件时,make会增加一个"-DEXPORT_SYMTAB"编绎标志,它使源文件嵌 入modversions.h文件,将EXPORT_SYMBOL宏展开中的函数名字符串进行版本名扩展;同时,它也定义_set_ver()宏为一空操 作,使代码中的函数名不受其影响。在编绎模块时,make会增加"-include=linux/modversion.h -DMODVERSIONS"编绎标志,使模块中代码的函数名得到相应版本扩展。由于生成.ver文件比较费时,make还为每个.ver创建了一个后缀为.stamp时戳文件。在make dep时,如果其.stamp文件比源文件旧才重新生成.ver文件,否则只是更新.stamp文件时戳。另外,在生成.ver和 modversions.h文件时,make都会比较新文件和旧文件的内容,保持它们修改时间为最旧。==========================================Makefile解读之四: Rules.make的注释==========================================## This file contains rules which are shared between multiple Makefiles.### False targets.##.PHONY: dummy## Special variables which should not be exported## 取消这些变量通过环境向make子进程传递。unexport EXTRA_AFLAGS
# as 的开关unexport EXTRA_CFLAGS
# cc 的开关unexport EXTRA_LDFLAGS
# ld 的开关unexport EXTRA_ARFLAGS
# ar 的开关unexport SUBDIRS
#unexport SUB_DIRS
# 编绎内核需进入的子目录,等于subdir-yunexport ALL_SUB_DIRS
# 所有的子目录unexport MOD_SUB_DIRS
# 编绎模块需进入的子目录unexport O_TARGET
# ld合并的输出对象unexport ALL_MOBJS
# 所有的模块名unexport obj-y
# 编绎成内核的文件集unexport obj-m
# 编绎成模块的文件集unexport obj-n
#unexport obj-
#unexport export-objs
# 需进行版本处理的文件集unexport subdir-y
# 编绎内核所需进入的子目录unexport subdir-m
# 编绎模块所需进入的子目录unexport subdir-nunexport subdir-## Get things started.#first_rule: sub_dirs
$(MAKE) all_targets# 在内核编绎子目录中过滤出可以作为模块的子目录。both-m
:= $(filter $(mod-subdirs), $(subdir-y))SUB_DIRS
:= $(subdir-y)# 求出总模块子目录MOD_SUB_DIRS
:= $(sort $(subdir-m) $(both-m))# 求出总子目录ALL_SUB_DIRS
:= $(sort $(subdir-y) $(subdir-m) $(subdir-n) $(subdir-))## Common rules## 将c文件编绎成汇编文件的规则,$@为目标对象。%.s: %.c
$(CC) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) -S $< -o $@# 将c文件生成预处理文件的规则。%.i: %.c
$(CPP) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) $< >; $@# 将c文件编绎成目标文件的规则,$<为第一个所依赖的对象;#在目标文件的目录下生成flags文件,strip删除多余的空格,subst将逗号替换成冒号。%.o: %.c
$(CC) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) -c -o $@ $<
@ ( \
echo 'ifeq ($(strip $(subst $(comma),:,$(CFLAGS) $(EXTRA_CFLAGS)$(CFLAGS_$@))),$$(strip $$(subst $$(comma),:,$$(CFLAGS) $$(EXTRA_CFLAGS)$$(CFLAGS_$@))))' ; \
echo 'FILES_FLAGS_UP_TO_DATE += $@' ; \
echo 'endif' \
) >; $(dir $@)/.$(notdir $@).flags# 汇编文件生成目标文件的规则。%.o: %.s
$(AS) $(AFLAGS) $(EXTRA_CFLAGS) -o $@ $<# Old makefiles define their own rules for compiling .S files,# but these standard rules are available for any Makefile that# wants to use them.
Our plan is to incrementally convert all# the Makefiles to these standard rules.
-- rmk, mecifdef USE_STANDARD_AS_RULE# 汇编文件生成预处理文件的标准规则。%.s: %.S
$(CPP) $(AFLAGS) $(EXTRA_AFLAGS) $(AFLAGS_$@) $< >; $@# 汇编文件生成目标文件的标准规则。%.o: %.S
$(CC) $(AFLAGS) $(EXTRA_AFLAGS) $(AFLAGS_$@) -c -o $@ $; $(dir $@)/.$(notdir $@).flagsendif # O_TARGET## Rule to compile a set of .o files into one .a file## 将obj-y组合成库L_TARGET的方法。ifdef L_TARGET$(L_TARGET): $(obj-y)
rm -f $@
$(AR) $(EXTRA_ARFLAGS) rcs $@ $(obj-y)
@ ( \
echo 'ifeq ($(strip $(subst $(comma),:,$(EXTRA_ARFLAGS)$(obj-y))),$$(strip $$(subst $$(comma),:,$$(EXTRA_ARFLAGS) $$(obj-y))))' ;\
echo 'FILES_FLAGS_UP_TO_DATE += $@' ; \
echo 'endif' \
) >; $(dir $@)/.$(notdir $@).flagsendif## This make dependencies quickly## wildcard为查找目录中的文件名的宏。fastdep: dummy
$(TOPDIR)/scripts/mkdep $(wildcard *.[chS] local.h.master) >; .dependifdef ALL_SUB_DIRS#将ALL_SUB_DIRS中的目录名加上前缀_sfdep_作为目标运行子make,并将ALL_SUB_DIRS通过# 变量_FASTDEP_ALL_SUB_DIRS传递给子make。
$(MAKE) $(patsubst %,_sfdep_%,$(ALL_SUB_DIRS))_FASTDEP_ALL_SUB_DIRS="$(ALL_SUB_DIRS)"endififdef _FASTDEP_ALL_SUB_DIRS#与上一段相对应,定义子目录目标,并将目标名还原为目录名,进入该子目录make。$(patsubst %,_sfdep_%,$(_FASTDEP_ALL_SUB_DIRS)):
$(MAKE) -C $(patsubst _sfdep_%,%,$@) fastdependif
## A rule to make subdirectories## 下面2段完成内核编绎子目录中的make。subdir-list = $(sort $(patsubst %,_subdir_%,$(SUB_DIRS)))sub_dirs: dummy $(subdir-list)ifdef SUB_DIRS$(subdir-list) : dummy
$(MAKE) -C $(patsubst _subdir_%,%,$@)endif## A rule to make modules## 求出有效的模块文件表。ALL_MOBJS = $(filter-out $(obj-y), $(obj-m))ifneq "$(strip $(ALL_MOBJS))" ""# 取主目录TOPDIR到当前目录的路径。PDWN=$(shell $(CONFIG_SHELL) $(TOPDIR)/scripts/pathdown.sh)endifunexport MOD_DIRSMOD_DIRS := $(MOD_SUB_DIRS) $(MOD_IN_SUB_DIRS)# 编绎模块时,进入模块子目录的方法。ifneq "$(strip $(MOD_DIRS))" "".PHONY: $(patsubst %,_modsubdir_%,$(MOD_DIRS))$(patsubst %,_modsubdir_%,$(MOD_DIRS)) : dummy
$(MAKE) -C $(patsubst _modsubdir_%,%,$@) modules# 安装模块时,进入模块子目录的方法。.PHONY: $(patsubst %,_modinst_%,$(MOD_DIRS))$(patsubst %,_modinst_%,$(MOD_DIRS)) : dummy
$(MAKE) -C $(patsubst _modinst_%,%,$@) modules_installendif# make modules 的入口。.PHONY: modulesmodules: $(ALL_MOBJS) dummy \
$(patsubst %,_modsubdir_%,$(MOD_DIRS)).PHONY: _modinst__# 拷贝模块的过程。_modinst__: dummyifneq "$(strip $(ALL_MOBJS))" ""
mkdir -p $(MODLIB)/kernel/$(PDWN)
cp $(ALL_MOBJS) $(MODLIB)/kernel/$(PDWN)endif# make modules_install 的入口,进入子目录安装。.PHONY: modules_installmodules_install: _modinst__ \
$(patsubst %,_modinst_%,$(MOD_DIRS))## A rule to do nothing#dummy:## This is useful for testing#script:
$(SCRIPT)## This sets version suffixes on exported symbols# Separate the object into "normal" objects and "exporting" objects# Exporting objects are: all objects that define symbol tables#ifdef CONFIG_MODULES# list-multi列出那些由多个文件复合而成的模块;# 从编绎文件表和模块文件表中过滤出复合模块名。multi-used
:= $(filter $(list-multi), $(obj-y) $(obj-m))# 取复合模块的构成表。multi-objs
:= $(foreach m, $(multi-used), $($(basename $(m))-objs))# 求出需进行编译的总模块表。active-objs
:= $(sort $(multi-objs) $(obj-y) $(obj-m))ifdef CONFIG_MODVERSIONSifneq "$(strip $(export-objs))" ""# 如果有需要进行版本化的文件。MODINCL = $(TOPDIR)/include/linux/modules# The -w option (enable warnings) for genksyms will return here in 2.1# So where has it gone?## Added the SMP separator to stop module accidents between uniprocessor# and SMP Intel boxes - AC - from bits by Michael Chastain#ifdef CONFIG_SMP
genksyms_smp_prefix := -p smp_else
genksyms_smp_prefix :=endif# 从源文件计算版本文件的规则。$(MODINCL)/%.ver: %.c
@if [ ! -r $(MODINCL)/$*.stamp -o $(MODINCL)/$*.stamp -ot $< ]; then \
echo '$(CC) $(CFLAGS) -E -D__GENKSYMS__ $<'; \
echo ' $(GENKSYMS) $(genksyms_smp_prefix) -k$(VERSION).$(PATCHLEVEL).$(SUBLEVEL) >; $@.tmp'; \
$(CC) $(CFLAGS) -E -D__GENKSYMS__ $< \
$(GENKSYMS) $(genksyms_smp_prefix) -k$(VERSION).$(PATCHLEVEL).$(SUBLEVEL) >; $@.tmp; \
if [ -r $@ ] && cmp -s $@ $@.tmp; then echo $@ is unchanged; rm -f$@.tmp; \
else echo mv $@.tmp $@; mv -f $@.tmp $@; fi; \
fi; touch $(MODINCL)/$*.stamp#将版本处理源文件的扩展名改为.ver,并加上完整的路径名,它们依赖于autoconf.h?br>;?br>;$(addprefix $(MODINCL)/,$(export-objs:.o=.ver)):$(TOPDIR)/include/linux/autoconf.h# updates .ver files but not modversions.h# 通过fastdep,逐个生成export-objs对应的版本文件。fastdep: $(addprefix $(MODINCL)/,$(export-objs:.o=.ver))# updates .ver files and modversions.h like before (is this needed?)# make dep过程的入口dep: fastdep update-modverfileendif # export-objs# update modversions.h, but only if it would change# 刷新版本文件的过程。update-modverfile:
@(echo "#ifndef _LINUX_MODVERSIONS_H";\
echo "#define _LINUX_MODVERSIONS_H"; \
echo "#include ;"; \
cd $(TOPDIR)/include/linux/modules; \
for f in *.ver; do \
if [ -f $$f ]; then echo "#include ;"; fi; \
done; \
echo "#endif"; \
) >; $(TOPDIR)/include/linux/modversions.h.tmp
@if [ -r $(TOPDIR)/include/linux/modversions.h ] && cmp -s$(TOPDIR)/include/linux/modversions.h$(TOPDIR)/include/linux/modversions.h.tmp; then \
echo $(TOPDIR)/include/linux/modversions.h was not updated; \
rm -f $(TOPDIR)/include/linux/modversions.h.tmp; \
else \
echo $(TOPDIR)/include/linux/modversions.h was updated; \
mv -f $(TOPDIR)/include/linux/modversions.h.tmp$(TOPDIR)/include/linux/modversions.h; \
fi$(active-objs): $(TOPDIR)/include/linux/modversions.helse# 如果没有配置版本化,modversions.h的内容。$(TOPDIR)/include/linux/modversions.h:
@echo "#include ;" >; $@endif # CONFIG_MODVERSIONSifneq "$(strip $(export-objs))" ""# 版本化目标文件的编绎方法。$(export-objs): $(export-objs:.o=.c) $(TOPDIR)/include/linux/modversions.h
$(CC) $(CFLAGS) $(EXTRA_CFLAGS) $(CFLAGS_$@) -DEXPORT_SYMTAB -c $(@:.o=.c)
@ ( \
echo 'ifeq ($(strip $(subst $(comma),:,$(CFLAGS) $(EXTRA_CFLAGS)$(CFLAGS_$@) -DEXPORT_SYMTAB)),$$(strip $$(subst $$(comma),:,$$(CFLAGS)$$(EXTRA_CFLAGS) $$(CFLAGS_$@) -DEXPORT_SYMTAB)))' ; \
echo 'FILES_FLAGS_UP_TO_DATE += $@' ; \
echo 'endif' \
) >; $(dir $@)/.$(notdir $@).flagsendifendif # CONFIG_MODULES## include dependency files if they exist## 嵌入源文件之间的依赖关系。ifneq ($(wildcard .depend),)include .dependendif# 嵌入头文件之间的依赖关系。ifneq ($(wildcard $(TOPDIR)/.hdepend),)include $(TOPDIR)/.hdependendif## Find files whose flags have changed and force recompilation.# For safety, this works in the converse direction:#
every file is forced, except those whose flags are positivelyup-to-date.## 已经更新过的文件列表。FILES_FLAGS_UP_TO_DATE :=# For use in expunging commas from flags, which mung our checking.comma = ,# 将当前目录下所有flags文件嵌入。FILES_FLAGS_EXIST := $(wildcard .*.flags)ifneq ($(FILES_FLAGS_EXIST),)include $(FILES_FLAGS_EXIST)endif# 将无需更新的文件从总的对象中删除。FILES_FLAGS_CHANGED := $(strip \
$(filter-out $(FILES_FLAGS_UP_TO_DATE), \
$(O_TARGET) $(L_TARGET) $(active-objs) \
))# A kludge: .S files don't get flag dependencies (yet),#
because that will involve changing a lot of Makefiles.
Also#
suppress object files explicitly listed in $(IGNORE_FLAGS_OBJS).#
This allows handling of assembly files that get translated into#
multiple object files (see arch/ia64/lib/idiv.S, for example).## 将由汇编文件生成的目件文件从FILES_FLAGS_CHANGED删除。FILES_FLAGS_CHANGED := $(strip \
$(filter-out $(patsubst %.S, %.o, $(wildcard *.S)$(IGNORE_FLAGS_OBJS)), \
$(FILES_FLAGS_CHANGED)))# 将FILES_FLAGS_CHANGED设为目标。ifneq ($(FILES_FLAGS_CHANGED),)$(FILES_FLAGS_CHANGED): dummyendif/////////////////////////////////////////////////////////////////////////////////////////////////////////////////如何编译Linux的内核姓名 :张子波 Email: 地址 : 转摘 :www.ccidnet.com
内核,是一个操作系统的核心。它负责管理系统的进程、内存、设备驱动程序、文件和网络系统,决定着系统的性能和稳定性。Linux作为一个自由软件,在广大爱好者的支持下,内核版本不断更新。新的内核修订了旧内核的bug,并增加了许多新的特性。如果用户想要使用这些新特性,或想根据自己的系统度身定制一个更高效,更稳定的内核,就需要重新编译内核。本文将以RedHat Linux 6.0(kernel 2.2.5)为操作系统平台,介绍在Linux上进行内核编译的方法。
一、 下载新内核的源代码
目前,在Internet上提供Linux源代码的站点有很多,读者可以选择一个速度较快的站点下载。笔者是从站点www.kernelnotes.org上下载了Linux的最新开发版内核2.3.14的源代码,全部代码被压缩到一个名叫Linux-2.3.14.tar.gz的文件中。
二、 释放内核源代码
由于源代码放在一个压缩文件中,因此在配置内核之前,要先将源代码释放到指定的目录下。首先以root帐号登录,然后进入/usr/src子目录。如果用户在安装Linux时,安装了内核的源代码,则会发现一个linux-2.2.5的子目录。该目录下存放着内核2.2.5的源代码。此外,还会发现一个指向该目录的链接linux。删除该连接,然后将新内核的源文件拷贝到/usr/src目录中。
(一)、用tar命令释放内核源代码
# cd /usr/src
# tar zxvf Linux-2.3.14.tar.gz
文件释放成功后,在/usr/src目录下会生成一个linux子目录。其中包含了内核2.3.14的全部源代码。
(二)、将/usr/include/asm、/usr/inlude/linux、/usr/include/scsi链接到/usr/src/linux/include目录下的对应目录中。
# cd /usr/include
# rm -Rf asm linux
# ln -s /usr/src/linux/include/asm-i386 asm
# ln -s /usr/src/linux/include/linux linux
# ln -s /usr/src/linux/include/scsi scsi
(三)、删除源代码目录中残留的.o文件和其它从属文件。
# cd /usr/src/linux
# make mrproper
三、 配置内核
(一)、启动内核配置程序。
# cd /usr/src/linux
# make config
除了上面的命令,用户还可以使用make menuconfig命令启动一个菜单模式的配置界面。如果用户安装了X window系统,还可以执行make xconfig命令启动X window下的内核配置程序。
(二)、配置内核
Linux的内核配置程序提供了一系列配置选项。对于每一个配置选项,用户可以回答"y"、"m"或"n"。其中"y"表示将相应特性的支持或设备驱动程序编译进内核;"m"表示将相应特性的支持或设备驱动程序编译成可加载模块,在需要时,可由系统或用户自行加入到内核中去;"n"表示内核不提供相应特性或驱动程序的支持。由于内核的配置选项非常多,本文只介绍一些比较重要的选项。
1、Code maturity level options(代码成熟度选项)
Prompt for development and/or incomplete code/drivers (CONFIG_EXPERIMENTAL) [N/y/?] 如果用户想要使用还处于测试阶段的代码或驱动,可以选择“y”。如果想编译出一个稳定的内核,则要选择“n”。
1、 Processor type and features(处理器类型和特色)
(1)、Processor family (386, 486/Cx486, 586/K5/5x86/6x86, Pentium/K6/TSC, PPro/6x86MX) [PPro/6x86MX] 选择处理器类型,缺省为Ppro/6x86MX。
(2)、Maximum Physical Memory (1GB, 2GB) [1GB] 内核支持的最大内存数,缺省为1G。
(3)、Math emulation (CONFIG_MATH_EMULATION) [N/y/?] 协处理器仿真,缺省为不仿真。
(4)、MTRR (Memory Type Range Register) support (CONFIG_MTRR) [N/y/?]
选择该选项,系统将生成/proc/mtrr文件对MTRR进行管理,供X server使用。
(5)、Symmetric multi-processing support (CONFIG_SMP) [Y/n/?] 选择“y”,内核将支持对称多处理器。
2、 Loadable module support(可加载模块支持)
(1)、Enable loadable module support (CONFIG_MODULES) [Y/n/?] 选择“y”,内核将支持加载模块。
(2)、Kernel module loader (CONFIG_KMOD) [N/y/?] 选择“y”,内核将自动加载那些可加载模块,否则需要用户手工加载。
3、 General setup(一般设置)
(1)、Networking support (CONFIG_NET) [Y/n/?] 该选项设置是否在内核中提供网络支持。
(2)、PCI support (CONFIG_PCI) [Y/n/?] 该选项设置是否在内核中提供PCI支持。
(3)、PCI access mode (BIOS, Direct, Any) [Any] 该选项设置Linux探测PCI设备的方式。选择“BIOS”,Linux将使用BIOS;选择“Direct”,Linux将不通过BIOS;选择 “Any”,Linux将直接探测PCI设备,如果失败,再使用BIOS。
(4)Parallel port support (CONFIG_PARPORT) [N/y/m/?] 选择“y”,内核将支持平行口。
4、 Plug and Play configuration(即插即用设备支持)
(1)、Plug and Play support (CONFIG_PNP) [Y/m/n/?] 选择“y”,内核将自动配置即插即用设备。
(2)、ISA Plug and Play support (CONFIG_ISAPNP) [Y/m/n/?] 选择“y”,内核将自动配置基于ISA总线的即插即用设备。
5、 Block devices(块设备)
(1)、Normal PC floppy disk support (CONFIG_BLK_DEV_FD) [Y/m/n/?] 选择“y”,内核将提供对软盘的支持。
(2)、Enhanced IDE/MFM/RLL disk/cdrom/tape/floppy support (CONFIG_BLK_DEV_IDE) [Y/m/n/?] 选择“y”,内核将提供对增强IDE硬盘、CDROM和磁带机的支持。
6、 Networking options(网络选项)
(1)、Packet socket (CONFIG_PACKET) [Y/m/n/?] 选择“y”,一些应用程序将使用Packet协议直接同网络设备通讯,而不通过内核中的其它中介协议。
(2)、Network firewalls (CONFIG_FIREWALL) [N/y/?] 选择“y”,内核将支持防火墙。
(3)、TCP/IP networking (CONFIG_INET) [Y/n/?] 选择“y”,内核将支持TCP/IP协议。
(4)The IPX protocol (CONFIG_IPX) [N/y/m/?] 选择“y”,内核将支持IPX协议。
(5)、Appletalk DDP (CONFIG_ATALK) [N/y/m/?] 选择“y”,内核将支持Appletalk DDP协议。
8、SCSI support(SCSI支持)
如果用户要使用SCSI设备,可配置相应选项。
9、Network device support(网络设备支持)
Network device support (CONFIG_NETDEVICES) [Y/n/?] 选择“y”,内核将提供对网络驱动程序的支持。
10、Ethernet (10 or 100Mbit)(10M或100M以太网)
在该项设置中,系统提供了许多网卡驱动程序,用户只要选择自己的网卡驱动就可以了。此外,用户还可以根据需要,在内核中加入对FDDI、PPP、SLIP和无线LAN(Wireless LAN)的支持。
11、Character devices(字符设备)
(1)、Virtual terminal (CONFIG_VT) [Y/n/?] 选择“y”,内核将支持虚拟终端。
(2)、Support for console on virtual terminal (CONFIG_VT_CONSOLE) [Y/n/?]
选择“y”,内核可将一个虚拟终端用作系统控制台。
(3)、Standard/generic (dumb) serial support (CONFIG_SERIAL) [Y/m/n/?]
选择“y”,内核将支持串行口。
(4)、Support for console on serial port (CONFIG_SERIAL_CONSOLE) [N/y/?]
选择“y”,内核可将一个串行口用作系统控制台。
12、Mice(鼠标)
PS/2 mouse (aka "auxiliary device") support (CONFIG_PSMOUSE) [Y/n/?] 如果用户使用的是PS/2鼠标,则该选项应该选择“y”。
13、Filesystems(文件系统)
(1)、Quota support (CONFIG_QUOTA) [N/y/?] 选择“y”,内核将支持磁盘限额。
(2)、Kernel automounter support (CONFIG_AUTOFS_FS) [Y/m/n/?] 选择“y”,内核将提供对automounter的支持,使系统在启动时自动 mount远程文件系统。
(3)、DOS FAT fs support (CONFIG_FAT_FS) [N/y/m/?] 选择“y”,内核将支持DOS FAT文件系统。
(4)、ISO 9660 CDROM filesystem support (CONFIG_ISO9660_FS) [Y/m/n/?]
选择“y”,内核将支持ISO 9660 CDROM文件系统。
(5)、NTFS filesystem support (read only) (CONFIG_NTFS_FS) [N/y/m/?]
选择“y”,用户就可以以只读方式访问NTFS文件系统。
(6)、/proc filesystem support (CONFIG_PROC_FS) [Y/n/?] /proc是存放Linux系统运行状态的虚拟文件系统,该项必须选择“y”。
(7)、Second extended fs support (CONFIG_EXT2_FS) [Y/m/n/?] EXT2是Linux的标准文件系统,该项也必须选择“y”。
14、Network File Systems(网络文件系统)
(1)、NFS filesystem support (CONFIG_NFS_FS) [Y/m/n/?] 选择“y”,内核将支持NFS文件系统。
(2)、SMB filesystem support (to mount WfW shares etc.) (CONFIG_SMB_FS)
选择“y”,内核将支持SMB文件系统。
(3)、NCP filesystem support (to mount NetWare volumes) (CONFIG_NCP_FS)
选择“y”,内核将支持NCP文件系统。
15、Partition Types(分区类型)
该选项支持一些不太常用的分区类型,用户如果需要,在相应的选项上选择“y”即可。
16、Console drivers(控制台驱动)
VGA text console (CONFIG_VGA_CONSOLE) [Y/n/?] 选择“y”,用户就可以在标准的VGA显示方式下使用Linux了。
17、Sound(声音)
Sound card support (CONFIG_SOUND) [N/y/m/?] 选择“y”,内核就可提供对声卡的支持。
18、Kernel hacking(内核监视)
Magic SysRq key (CONFIG_MAGIC_SYSRQ) [N/y/?] 选择“y”,用户就可以对系统进行部分控制。一般情况下选择“n”。
四、 编译内核
(一)、建立编译时所需的从属文件
# cd /usr/src/linux
# make dep
(二)、清除内核编译的目标文件
# make clean
(三)、编译内核
# make zImage
内核编译成功后,会在/usr/src/linux/arch/i386/boot目录中生成一个新内核的映像文件zImage。如果编译的内核很大的话,系统会提示你使用make bzImage命令来编译。这时,编译程序就会生成一个名叫bzImage的内核映像文件。
(四)、编译可加载模块
如果用户在配置内核时设置了可加载模块,则需要对这些模块进行编译,以便将来使用insmod命令进行加载。
# make modules
# make modelus_install
编译成功后,系统会在/lib/modules目录下生成一个2.3.14子目录,里面存放着新内核的所有可加载模块。
五、 启动新内核
(一)、将新内核和System.map文件拷贝到/boot目录下
# cp /usr/src/linux/arch/i386/boot/bzImage /boot/vmlinuz-2.3.14
# cp /usr/src/linux/System.map /boot/System.map-2.3.14
# cd /boot
# rm -f System.map
# ln -s System.map-2.3.14 System.map
(二)、配置/etc/lilo.conf文件。在该文件中加入下面几行:
default=linux-2.3.14
image=/boot/vmlinuz-2.3.14
label=linux-2.3.14
root=/dev/hda1
read-only
(三)、使新配置生效
# /sbin/lilo
(四)、重新启动系统
# /sbin/reboot
新内核如果不能正常启动,用户可以在LILO:提示符下启动旧内核。然后查出故障原因,重新编译新内核即可。//////////////////////////////////////////////////////////////////////
相关信息· Spring中事件处理的小技巧
· Linux文件基本属性
· 什么是ERP中的E
· linux下通过蓝牙手机上网
97226
88182
