背景

项目中需要处理Android的原生开机动画,一定条件下还需要做到静默重启(android系统启动进入到桌面前,屏幕保持完全没有亮度的状态)。因为项目是基于Android Q做的,顺势探索了一波开机动画的细节。

开机动画到底有几个

搜了一些资料,推荐一篇博客,原理讲的很细,汇总开机动画一共有三个:
Android系统的开机画面显示过程分析 总结下:
1、Linux内核的启动画面,这个动画一般不显示。
2、第二个开机画面是在init进程启动的过程中出现的,它也是一个静态的画面。以小米手机为例,开机出现的第一个白色的 “MI” logo。
3、第三个开机动画一般各手机厂商会自行定制,也就是开机后的动画特效。
但是基于Q测试时,还发现一个“Phone is starting”的对话框页面,这个页面后才是桌面。于是引入了第四个开机动画:
4、启动Launcher之前会先启动一个FallbackHome,这个FallbackHome就是“Phone is starting”的对话框页面,这个页面一般厂商也会自行定制。
综上:目前最新的Android源码一共有四个开机动画。

自定义开机动画

一、定制第一个、第二个开机动画

这个基本没有需求,我没研究,有需求的同学可以参考Android系统的开机画面显示过程分析学习研究。

二、定制第三个开机动画

Android系统默认的开机动画是由两张图片android-logo-mask.png和android-logo-shine.png中。这两张图片保存在frameworks/base/core/res/assets/images目录中。一般厂商会自行定制此动画,谷歌也提供了替换的方法:

#define USER_BOOTANIMATION_FILE "/data/local/bootanimation.zip"
#define SYSTEM_BOOTANIMATION_FILE "/system/media/bootanimation.zip"

替换两个目录下的任何一个即可。一般会在rom源码去替换这个文件,示例代码:
需要修改device.mk(在rom根目录的device目录中)文件,同时把自定义的bootanimation.zip放到device.mk所在git库根目录

PRODUCT_COPY_FILES += $(LOCAL_PATH)/bootanimation.zip:system/media/bootanimation.zip

注: 如果需要隐藏此动画,有个取巧的方法,可以bootanimation.zip内放不合适的image图片,就不会显示此动画了。相比改代码处理更简洁。

三、定制第四个开机动画

也就是改动src/com/android/settings/FallbackHome.java这个类。这个类是在系统应用Settings中
项目路径是:rom根目录/packages/apps/Settings,改动很简单,直接把显示dialog的代码注掉(或者自行定制)

@Override
    protected void onResume() {
        super.onResume();
        if (mProvisioned) {
            // mHandler.postDelayed(mProgressTimeoutRunnable, PROGRESS_TIMEOUT);
        }

如果系统已经做过开机引导,不会显示此动画。
注:验收过程中,mtk平台的rom自行定制了Settings项目,改动/packages/apps/Settings无效,MTK平台下的rom,需要改动rom根目录/vendor/mediatek/proprietary/packages/apps/MtkSettings下的FallbackHome.java**

静默重启,屏幕亮度为为0

分析完上边的动画,如果需要在设备启动中变更屏幕亮度。一共需要处理两大块。
1、修改第二个动画的屏幕背光。
2、修改第三个动画的屏幕背光。

一、在misc分区记录亮度值

因为第二个开机动画,整个android的文件系统还没启动,所以无法使用文件系统记录。所以使用misc分区记录。
这里大概解释下:
android上主要有六个分区:cache,dumisc,recovery,boot,system,userdata。
后四种分区不做解释了,主要在于misc和cache两个分区的作用。cache是缓存空间,程序和系统用到的缓存数据和指令就存放在这里;CPU在调用和执行命令是会优先调用这里的数据;misc分区中有Bootloader Control Block(BCB),主要是用于存放Recovery引导信息。第二个开机动画(静态logo)就需要借用此分区。
1、从misc分区读亮度值,并设置。
设置背光值的源码位置:
vendor/mediatek/proprietary/bootable/bootloader/lk/platform/mt…(型号代号)/mt_leds.c

void mt65xx_backlight_on(void)
{
	// backlight_level增加逻辑,从misc分区读出背光值,在这里设置即可
	enum led_brightness backlight_level = get_cust_led_default_level();
	LEDS_INFO("[LEDS]LK: mt65xx_backlight_on:level =  %d\n\r",backlight_level);
	mt65xx_leds_brightness_set(MT65XX_LED_TYPE_LCD, backlight_level);
	LEDS_INFO("[LEDS]LK: mt65xx_leds_brightness_set is done\n");
}

2、暴露给app可以操作misc分区此值的工具。
app如果想修改misc分区数据,需要提供c程序,app可以使用命令调用。c程序保密,不过原理大概这样。

二、修改第三个开机动画往后的背光

1、因为从这里开始文件系统已经加载,所以使用文件记录是否设置背光为0。
2、本以为第三个开机动画到桌面显示,也就一个地方设置了背光,代码尝试改动了
frameworks/base/services/core/java/com/android/server/display/DisplayPowerController.java

private void updatePowerState() {
        ......
        if (brightness < 0 && mPowerRequest.screenBrightnessOverride > 0) {
            // 这里使用/data/backlight_0文件是否存在标记
            File file = new File("/data/backlight_0");
            brightness = file.exists() ? 0 : mPowerRequest.screenBrightnessOverride;
            mBrightnessReasonTemp.setReason(BrightnessReason.REASON_OVERRIDE);
            mAppliedScreenBrightnessOverride = true;
        } else {
            mAppliedScreenBrightnessOverride = false;
        }
        ......
  }

改动后发现并没有把整个流程的背光设置为0。
3、后来发现整个过程多个地方操作了背光,总不能所有的地方都去变动。于是找到了设置背光的源头:
vendor/mediatek/proprietary/hardware/liblights/lights.c
增加判断:

static int
write_int(char const* path, int value)
{
    int fd;

#ifdef LIGHTS_INFO_ON
	ALOGD("write %d to %s", value, path);
#endif

	// 此处增加了逻辑,如果文件存在直接返回。
    if ((access("/data/backlight_0", F_OK)) != -1) {
        return 0;
    }

    fd = open(path, O_RDWR);
	ALOGD("write_int open fd=%d\n", fd);
    if (fd >= 0) {
        char buffer[20];
        int bytes = sprintf(buffer, "%d\n", value);
        int amt = write(fd, buffer, bytes);
        close(fd);
        return amt == -1 ? -errno : 0;
    } else {
        return -errno;
    }
}

注:这里解释下此处设置背光的原理。是通过write写/sys/class/leds/lcd-backlight/brightness文件设置背光的。

三、设备进入到桌面后需要清掉自己定制的背光(毕竟特殊场景下的需求)

这里可能很多人想在桌面去改动。其实有个更好的地方!!!就是第四个动画的FallbackHome.java。这个要比桌面启动早。且更独立。并且还有最重要的一点,这个app是systemUid,可以操作更广的文件系统,权限也更高。
代码变动:

@Override
    protected void onPause() {
        super.onPause();
        mHandler.removeCallbacks(mProgressTimeoutRunnable);

        clearMyBackLight();
    }

    private void clearMyBackLight() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    // 1、需要执行提供的c程序,清掉第二个动画,也就是misc分区存储的值
                    Runtime.getRuntime().exec("/system/bin/提供的程序 clear");
                    // 2、清掉backlight_0,恢复第三个动画往后的正常屏幕背光
                    Runtime.getRuntime().exec("rm /data/backlight_0");
                    // 3、恢复重启前的用户设置的背光值,这里用了Settings.Secure存储
                    int value = Settings.Secure.getInt(getApplicationContext().getContentResolver(), "pref_key_back_light");
                    String[] args = {"echo", String.valueOf(value), ">", "/sys/class/leds/lcd-backlight/brightness"};

                    Process process = Runtime.getRuntime().exec("sh");
                    OutputStream outputStream = process.getOutputStream();
                    DataOutputStream dataOutputStream = new DataOutputStream(outputStream);

                    String cmd = "";
                    for (String arg : args) {
                        cmd += arg + " ";
                    }
                    cmd += "\n";
                    dataOutputStream.writeBytes(cmd);
                    dataOutputStream.flush();
                    dataOutputStream.writeBytes("exit\n");
                    dataOutputStream.flush();
                } catch (Exception e) {
                    // ignore
                }
            }
        }).start();
    }

结束

一次比较神奇和艰难的探索之旅。