zoukankan      html  css  js  c++  java
  • 深入浅出

    本文转载自:

    4.3定制Android平台系统

    通常产品厂商在拿到Android源码后会在android源码基础上进行定制修改,以匹配适应自己的产品,从本节开始,我们从最原始的Android源码系统里一步一步定制出自己的Android系统。本节主要内容包含:根据Android源码,添加新产品编译项,定制系统启动界面和文字,定制系统启动动画和声音,定制系统桌面。

    4.3.1 添加新产品编译项

    Android系统的源代码是一个逻辑结构非常独立工程,在一套Android源码中可以编译出多个产品映像,在需要编译某一个产品系统时,只要通过lunch命令选择产品编译项即可。本节我们介绍如何在Android源码中创建新产品编译项并定制编译出该产品系统。

    在创建新产品编译项时,要先了解下面几个概念:

    Ø 目标产品:具体指某个最终用户买到的Android设备,如:iPhone5,乐PhoneS2,小米手机等。

    Ø 产品系列:开发手机的团队通常由同一团队打造,在研发出一款产品后,往往要继续在其基础上研发出新产品,新产品往往是在老产品的硬件或软件基础上做一些升级,这些产品们就是一个产品系列。比如:联想的乐Phone系列手机包含:乐PhoneS1和乐PhoneS2,他们同属于一个系列。

    Ø 目标设备:目标设备可以理解为手机主板,它是指手机设备硬件配置信息的集合体,每个手机产品都有设备硬件配置,一个设备硬件配置可能被不同产品使用,同一手机有高配置版本和低配置版本,如乐PhoneS2有512M RAM、8G Flash容量版本和1G RAM 、16G Flash容量版本。

    在Android编译系统中,每个编译项编译出一个产品系统,每个目标产品都对应一个目标设备,一个产品系列包含多个不同的产品,一个目标设备可能被多个产品配置使用。

    由前面描述可知,同一系列的新老产品之间可以存在“继承关系”,新产品是老产品的“子产品”,老产品是新产品的“父产品”,子产品可以复用父产品的特性,还可以重写、扩展父产品。如:老产品不支持NFC近距离通信技术,新产品支持NFC技术。同样,设备主板间也存在“继承关系”。

     

    图x-x 产品、设备与编译项关系图

     

    如图x-x所示,某一产品系列包含3个产品,2个目标设备,其中产品2继承了产品1,产品2 使用了设备2,它是基于产品1所使用的设备1的升级。产品3使用了和产品2一样的设备2,他们硬件配置一样,但是却不是同一产品,3个不同产品都对应一个产品编译项。

    在Android编译系统中,产品编译项相关配置文件都在device/<厂商名>/目录下。厂商的产品列表由AndroidProducts.mk文件定义,目标产品信息由<产品名>.mk定义,目标设备信息由BoardConfig.mk和AndroidBoard.mk定义。创建新产品的编译项就是创建上述几个mk文件的过程。

    1.      创建厂商目录

    不同的手机厂商对应device/下不同目录,在厂商目录下放置该厂商的产品相关信息,我们厂商名定义为mycompany。

    $ cd ~/android/android_source

    $ mkdir device/mycompany

    2.      在厂商目录下创建设备目录

    定义设备名为myphone。

    $ mkdir device/mycompany/myphone

    3.      添加新产品编译项配置文件,该配置文件在执行source build/envsetup.sh时,被加载执行

    $ vim device/mycompany/myphone/vendorsetup.sh

    在vendorsetup.sh文件时添加下面一条命令,用于向编译系统添加编译项,新添加的产品名为:myproduct,编译类型为eng。

    add_lunch_combo myproduct-eng

    注:add_lunch_combo命令是build/envsetup.sh脚本中定义的函数,表示将一个新产品编译项添加到lunch菜单里。

    4.      创建产品列表配置文件AndroidProducts.mk

    AndroidProducts.mk文件用于定义当前厂商所拥有的所有产品列表,每个产品都对应一个配置文件:

    $ vimdevice/mycompany/myphone/AndroidProducts.mk

    在产品列表配置文件中添加如下内容:

    PRODUCT_MAKEFILES :=

       $(LOCAL_DIR)/full_product.mk

    注:PRODUCT_MAKEFILES变量用于保存所有产品配置信息列表,$(LOCAL_DIR)表示当前目录,full_product.mk表示某一款产品的配置文件。

    5.      配置full_product.mk文件,定义产品的配置信息,添加如下内容:

    include build/target/product/languages_full.mk

    include build/target/product/full.mk

    # Discard inherited values and use our owninstead.

    PRODUCT_NAME := myproduct

    PRODUCT_DEVICE := myphone

    产品配置也可以和Java中的类一样被继承,通过inclulde命令可以将指定的文件包含进来,然后在后面可以对里面的内容进行重写。一般而言不同的产品产品名和设备名都不一样,在full_product.mk中对继承的full.mk中的产品名和设备名进行重写:PRODUCT_NAME为myproduct,PRODUCT_DEVICE为myphone。

    在full_product.mk文件中继承的languages_full.mk内容如下:

    @build/target/product/languages_full.mk

    PRODUCT_LOCALES := en_US fr_FR it_IT es_ES de_DEnl_NL cs_CZ pl_PL ja_JP zh_TW zh_CN ru_RU ko_KR nb_NO es_US da_DK el_GR tr_TRpt_PT pt_BR rm_CH sv_SE bg_BG ca_ES en_GB fi_FI hr_HR hu_HU in_ID iw_IL lt_LTlv_LV ro_RO sk_SK sl_SI sr_RS uk_UA vi_VN tl_PH

    该配置文件里表示的是当前产品系统里默认支持的本地语言,由上述配置信息可知,它基本包含了Android所支持的所有语言包。

    @build/target/product/full.mk

    PRODUCT_PACKAGES :=

       OpenWnn

       PinyinIME

        VoiceDialer

       libWnnEngDic

       libWnnJpnDic

       libwnndict

    # Additional settings used in all AOSP builds

    PRODUCT_PROPERTY_OVERRIDES :=

       keyguard.no_require_sim=true

       ro.com.android.dateformat=MM-dd-yyyy

       ro.com.android.dataroaming=true

       ro.ril.hsxpa=1

       ro.ril.gprsclass=10

    PRODUCT_COPY_FILES :=

       development/data/etc/apns-conf.xml:system/etc/apns-conf.xml

       development/data/etc/vold.conf:system/etc/vold.conf

    # Pick up some sounds - stick with the shortlist to save space

    # on smaller devices.

    $(call inherit-product,frameworks/base/data/sounds/OriginalAudio.mk)

    # Get the TTS language packs

    $(call inherit-product-if-exists,external/svox/pico/lang/all_pico_languages.mk)

    # Get a list of languages. We use the small listto save space

    # on smaller devices.

    $(call inherit-product,build/target/product/languages_small.mk)

    $(call inherit-product,build/target/product/generic.mk)

    # Overrides

    PRODUCT_NAME := full

    PRODUCT_BRAND := generic

    PRODUCT_DEVICE := generic

    PRODUCT_MODEL := Full Android

    继承的full.mk文件内容比较多,我们将主要的一些变量列出如表x-x所示。

    变量名

    作用

    使用方式

    PRODUCT_PACKAGES

    系统预置的模块列表,不仅仅只是Android应用程序,还可以包含库,可执行程序等

    直接将系统中要安装的模块名以空格隔开列出

    PRODUCT_PROPERTY_OVERRIDES

    系统设置的属性值

    将所有预设的属性以空格隔开列出,属性格式为:key-value

    PRODUCT_COPY_FILES

    要拷贝的文件

    将文件列表拷贝到文件系统中,文件格式为:源文件:目标文件

    PRODUCT_NAME

    产品名

    该产品名要和编译项中产品名一致

    PRODUCT_BRAND

    产品品牌

    PRODUCT_DEVICE

    产品对应的设备名

    该名字要和产品设备主板配置文件(BoardConfig.mk)所在目录名一致

    PRODUCT_MODEL

    总结:我们自己定义的full_product产品继承了build/target/product/目录下的full.mk和languages_full.mk,full.mk文件是Android系统定义的一个“通用产品”,languages_full.mk文件是全部语言包配置文件,这样,自己的产品full_product就具有了通用产品的特点并且支持全部语言包。

    6.      定义目标产品对应的设备配置文件AndroidBoard.mk和BoardConfig.mk

    同样的道理,我们可以继承使用通用设备配置文件:build/target/board/generic/目录下的AndroidBoard.mk和BoardConfig.mk文件。

    Ø 创建AndroidBoard.mk和BoardConfig.mk文件

    $ touch AndroidBoard.mk BoardConfig.mk

    Ø 添加AndoridBoard.mk的内容如下:

    @ device/mycompany/myphone/AndroidBoard.mk

    include build/target/board/generic/AndroidBoard.mk

    “继承”的父AndroidBoard.mk,其内容:

    @build/target/board/generic/AndroidBoard.mk

    LOCAL_PATH := $(call my-dir)

    file := $(TARGET_OUT_KEYLAYOUT)/tuttle2.kl           # Linux内核按键码布局文件

    ALL_PREBUILT += $(file)

    $(file) : $(LOCAL_PATH)/tuttle2.kl | $(ACP)

             $(transform-prebuilt-to-target)

    include $(CLEAR_VARS)

    LOCAL_SRC_FILES := tuttle2.kcm            # Android按键码映射文件

    include $(BUILD_KEY_CHAR_MAP)

    其实build/target/board/generic/AndroidBoard.mk文件里只是拷贝了按键映射文件和默认系统属性文件,我们可以将其内容直接拷贝到device/mycompany/myphone/AndroidBoard.mk中。

    Ø 添加BoardConfig.mk的内容如下:

    @ device/mycompany/myphone/BoardConfig.mk

    includebuild/target/board/generic/BoardConfig.mk

    “继承”的父BoardConfig.mk内容:

    @build/target/board/generic/BoardConfig.mk

    # config.mk

    #

    # Product-specific compile-time definitions.

    #

    # The generic product target doesn't have anyhardware-specific pieces.

    TARGET_NO_BOOTLOADER := true                  # 当前设备是否没有Bootloader

    TARGET_NO_KERNEL := true                            # 当前设备是否没有linux内核

    TARGET_CPU_ABI := armeabi                           # 当前设备支持的目标架构

    HAVE_HTC_AUDIO_DRIVER := true                   # 是否使用HTC的音频驱动

    BOARD_USES_GENERIC_AUDIO := true          # 是否使用通用音频技术

    # no hardware camera

    USE_CAMERA_STUB := true                    # 是否使用摄像头Stub

    通过BoardConfig.mk的信息可知,其实该文件就是定义了一些设备硬件相关的一些变量,这些变量用来裁剪系统的功能,决定Android系统可运行的体系构架。

    7.      根据需要定义产品默认属性和键值信息

    Android系统的属性服务类似于Windows的注册表,记录着系统的一些设置信息,我们可以在新产品中预定义一些属性值来设置自己产品。在Android编译系统中,属性都保存在xxx.prop文件中,在build/target/board/generic/system.prop中定义了默认的属性,我们可以在它基础上进行修改。

    复制属性文件:

    $ cp build/target/board/generic/system.prop  device/mycompany/myphone/

    在Android系统中,底层使用Linux内核来接收来自按键硬件上报的键值信息,上层处理用户按键的是Android的框架,二者之间通过两个键值布局文件来进行键值的映射。

    Ø Keylayout文件:按键布局文件,以kl后缀命名,该文件用来定义按键驱动里上报的键值号(数字)和Linux内核中通过event事件上报的键值(字符)之间的映射关系。kl文件要放在/system/usr/keylayout/目录下或/data/usr/keylayout/目录下。

    Ø KeyCharMap文件:键值字符映射文件,以kcm后缀命名,它用来将Linux内核上报来的键值(字符)进行转换,转换成Android系统里可以识别的键盘码或组合按键。kcm文件要放在/system/usr/keychars/目录下或/data/usr/keychars/目录下。

    上述两个按键映射文件使用按键驱动名作为其文件名,如果没有驱动名对应的布局文件,则使用/system/usr/keylayout/qwerty.kl和/system/usr/keychars/qwerty.kcm作为默认的按键映射文件。这两个文件名都通过AndroidBoard.mk文件负责拷贝和安装。

    如果我们要使用模拟器作为目标设备,只需要将源码build/target/board/generic/目录里的tuttole2.kl和tuttle2.kcm拷贝到AndroidBoard.mk所在的目录中即可。

    $ cp build/target/board/generic/tuttle2.kl  device/mycompany/myphone/tuttle2.kl

    $ cp build/target/board/generic/tuttle2.kcm  device/mycompany/myphone/tuttle2.kcm 

    如果想要自定义系统的物理按键与Android系统的按键映射关系,则需要在tuttle2.kl和tuttle2.kcm的基础上进行修改,然后再修改AndroidBoard.mk的内容:

    $ cp build/target/board/generic/tuttle2.kl  device/mycompany/myphone/<按键驱动名>.kl

    $ cp build/target/board/generic/tuttle2.kcm  device/mycompany/myphone/<按键驱动名>.kcm 

    修改device/mycompany/myphone/AndroidBoard.mk文件:

    LOCAL_PATH := $(call my-dir)

    file := $(TARGET_OUT_KEYLAYOUT)/<按键驱动名>.kl           # Linux内核按键码布局文件

    ALL_PREBUILT += $(file)

    $(file) : $(LOCAL_PATH)/<按键驱动名>.kl | $(ACP)

             $(transform-prebuilt-to-target)

    include $(CLEAR_VARS)

    LOCAL_SRC_FILES := <按键驱动名>.kcm                  #  Android按键码映射文件

    include $(BUILD_KEY_CHAR_MAP)

    注:kcm文件最终被编译系统的key_char_map.mk编译成xxx.kcm.bin的二进制形式,这是因为每个Android应用程序都要加载该按键映射文件,为了加快读取速度刻意而为之的。

    创建新产品编译项时创建的目录与文件结构如下:

    device/mycompany/                          # 厂商目录

    └── vendorsetup.sh           # 添加编译项命令文件

    └── myphone/                    # 设备名目录

    ├── AndroidBoard.mk                  # 设备属性和键值映射配置文件

    ├── AndroidProducts.mk             # 产品列表文件

    ├── BoardConfig.mk          # 设备硬件配置及目标架构配置文件

    ├── full_product.mk            # 目标产品配置文件

    ├── system.prop                          # 系统默认属性配置文件

    ├── tuttle2.kcm                            # Android系统键值映射文件

    ├── tuttle2.kl                      # Linux内核按键布局文件

    确认上述目录和文件创建没有问题了,执行Android编译步骤:sourcebuild/envsetup.sh,lunch选择myproduct-eng编译项。

    如果看到如下信息,说明我们已经添加新产品成功。

    ============================================

    PLATFORM_VERSION_CODENAME=REL

    PLATFORM_VERSION=2.3.6

    TARGET_PRODUCT=myproduct

    TARGET_BUILD_VARIANT=eng

    TARGET_SIMULATOR=false

    TARGET_BUILD_TYPE=release

    TARGET_BUILD_APPS=

    TARGET_ARCH=arm

    HOST_ARCH=x86

    HOST_OS=linux

    HOST_BUILD_TYPE=release

    BUILD_ID=GRK39F

    ============================================

    8.      常见问题

    Ø 问题1: lunch菜单里没有出现myproduct编译项

    原因及解决方法:在执行lunch之前,要执行source build/envsetup.sh命令,确认vendorsetup.sh文件存在及其内容正确无误。

    Ø 问题2:选择完lunch菜单里的编译项后,出错:

    *** No matches for product"myproduct".  Stop.

    ** Don't have a product spec for:'myproduct'

    ** Do you have the right repo manifest?

    原因及解决方法:编译系统找不到用户选择的编译项里的myproduct产品,确认AndroidProducts.mk文件里列出了myproduct产品的配置文件full_product.mk,并且full_product.mk文件中PRODUCT_NAME变量的值为产品名:myproduct

    Ø 问题3:选择完lunch菜单里的编译项后,出错:

    *** No config file found for TARGET_DEVICEmyphone.  Stop.

    ** Don't have a product spec for:'myproduct'

    ** Do you have the right repo manifest?

    原因及解决方法:编译系统找不到myproduct产品对应的设备myphone,确认myproduct产品的配置文件full_product.mk中PRODUCT_DEVICE变量的值为产品名:myphone,并且在device/mycompany/目录下创建了myphone的设备目录,在该目录下存在BoardConfig.mk文件。

    4.3.2 定制产品的意义及定制要点

    Android系统是一个完全开源的系统,我们可以通过修改Linux内核代码和Android源码,定制具有独特创意的产品系统,对于产品同质化非常严重的移动市场, Android系统的细节个性化定制也可以让用户眼前一亮。另外,一些产品明确要求要修改或增加一些个性化,如:默认的Android系统开机界面是一个黄嘴的小企鹅,在Android系统启动过程中是一个ANDROID字样的动画效果,厂商一般都要求自己产品开机界面是厂商Logo,开机动画是一个能动态、鲜明表现公司活力的动画效果,我们从本节开始介绍定制产品系统的实现技术。

    在整个开机过程中,屏幕上会出现三次内容,如图x-x 所示:

    Ø  Linux启动时画面,通常是个黄嘴的小企鹅

    Ø  Android系统init进程启动阶段画面,是“ANDROID”文字字样

    Ø  Android系统启动阶段动画,是滚动的ANDROID动画

     

    图 x-x 开机界面与Android动画

    定制系统开机动画

    【实验背景知识】

    Android的开机动画是由Linux本地守护程序bootanimation专门控制实现的,其代码在:frameworks/base/cmds/bootanimation/目录下,修改Android开机动画有两种方式:

    Ø 蒙板图片替换:

    替换frameworks/base/core/res/assets/images/目录下的两个图片文件:android-logo-mask.png和android-logo-shine.png。android-logo-mask.png是镂空蒙板图片,android-logo-shine.png是镂空蒙板后面的闪光png图片。两个图片通过叠加移动来达到动画效果。

    Ø 逐帧动画替换:

    在/data/local/或/system/media/目录创建bootanimation.zip文件,该压缩包文件里存放有逐帧动画及控制脚本。

    【实验组成】

    本实验分为两部分:蒙板图片替换实验和逐帧动画替换实验。

    【实验内容】

    分析Android系统的两种开机动画实现方式,制作并替换开机动画,最终在Android模拟器中运行定制开机动画的系统。

    【实验目的】

    通过实验,了解Android系统的两种开机动画实现方式,掌握如何定制产品的开机动画,并在Android模拟器中,运行定制开机动画的Android系统。

    【实验平台】

    拥有Android源码编译环境的Ubuntu操作系统(可以在Windows系统中虚拟Ubuntu系统)。

    【蒙板图片替换实验步骤】

    1.       使用PhotoShop等图像处理软件制作一张背景为黑色,中间镂空的png格式的图片,命名为:android-logo-mask.png,如图x-x所示。

     

    图x-x 制作镂空动画

    2.      将android-logo-mask.png拷贝到frameworks/base/core/res/assets/images/目录下替换Android默认的图片,为了防止源码不编译图片资源,将图片时间戳更新一下。

    $ cp android-logo-mask.png    ~/android/android_source/frameworks/base/core/res/assets/images/

    $ touch ~/android/android_source/frameworks/base/core/res/assets/images/android-logo-mask.png

    3.      重新编译Android的系统资源包framework-res.apk

    $ source build/envsetup.sh

    $ lunch generic-eng

    $ mmm frameworks/base/core/res/

    4.      生成新的system.img

    $ make snod

    5.      启动Android模拟器,实验效果如图x-x所示。

    $ ./run_emulator.sh

     

     

    图x-x 定制开机动画效果

    【逐帧动画替换实验步骤】

    1.      在/data/local/或/system/media/目录创建bootanimation.zip文件

    如果放在/data/local目录下,不需要编译Android源码,直接通过adb命令或文件管理软件拷贝到目录下即可,如果集成进Android系统中,则需要放在/system/media/目录下,这时要重新编译生成system.img映像。

    bootanimation.zip文件是直接由几个文件打包生成的,打包的格式是ZIP,打包时的压缩方式选择为存储。

    图x-x 压缩文件方式

    bootanimation.zip文件打包前的结构为:

    表x-x bootanimation.zip压缩包文件结构

    文件

    说明

    desc.txt

    动画属性描述文件

    part0/

    第一阶段动画图片的目录

    part1/

    第二阶段动画图片的目录

    其中part0和part1中的动画图片类似于电影胶片,两张图片之间变化较小,他们以固定的速度显示,从而产生动画效果,图片的大小和图片显示的时间控制由desc.txt文件说明。

    desc.txt文件内容为:

    480 250 15

    p 1 0 part0

    p 0 10 part1

    desc.txt文件的格式为:

     

    数据及说明

    图片属性

    320(图片宽)

    320(图片高)

    15(每秒显示帧数)

    第一阶段动画属性

    P(默写标志符)

    1(循环次数为1 )

    0(进入该阶段的间隔时间)

    part0(该阶段图片存放目录)

    第二阶段动画属性

    p(默写标志符)

    0(无限循环)

    10(进入该阶段的间隔时间)

    part1(该阶段图片存放目录)

    注:

    标识符:p 是必须的。

    循环次数:指该目录中图片循环显示的次数,0表示本阶段无限循环。

    每秒显示帧数:就是每秒显示的图片数量,决定每张图片显示的时间。

    阶段切换间隔时间:指的是该阶段结束后间隔多长时间显示下一阶段的图片,其单位是每张图片显示的时间。

    对应图片目录:就是该阶段动画的系列图片,以图片文件目录的顺序显示动画,而且图片的格式必须要为PNG。

    由于逐帧动画不太方便制做,我们直接使用光盘中:章节实验/第四章定制系统开机动画/bootanimation.zip文件作为演示。

    2.      如果bootanimation.zip放到/system/media/目录下,则重新编译生成system.img

    $ source build/envsetup.sh

    $ lunch generic-eng

    $ make snod

    3.      启动Android模拟器,查看动画效果,如图x-x和x-x所示。

    $ ./run_emulator.sh

     

     

    图x-x 第一阶段开机动画

     

     

    图x-x 第二阶段开机动画

    结论:通过实验看出,当我们使用逐帧动画时,蒙板动画就不播放了,这是因为Android系统只能使用一种启动动画方式,先判断是否使用了逐帧动画,如果没有使用逐帧动画时,才使用默认的蒙板动画。

  • 相关阅读:
    [置顶] 国外程序员推荐:每个程序员都应读的书
    Android入门(2) 基本控件介绍、4种布局
    Struts2标签之<s:select>
    百度地图API学习总结
    Environment
    SD卡中文件夹和文件的操作
    Webpack v4.8.3 快速入门指南
    javascript代码模块化解决方案
    MVC,MVP 和 MVVM 的图示 阮一峰
    webpack Uncaught ReferenceError: Swiper is not defined
  • 原文地址:https://www.cnblogs.com/zzb-Dream-90Time/p/7172167.html
Copyright © 2011-2022 走看看