Init的工作职责

注意
本文最后更新于 2024-01-20,文中内容可能已过时。

原文链接:https://ovea-y.cn/init_work_1/

Flowchart

ueventd其实就是init,为了方便启动所以单独设置了一个链接

Pasted image 20231112115509

Pasted image 20231112115516

它的作用初始化属性,它在非常早期的阶段(此时没有执行init脚本指令,甚至没有初始化selinux)就会执行。 它会做下面这些操作:

  • 创建/dev/__properties__节点
  • 解析序列化的属性(在文件{分区名}__property_contexts中),并且写入到/dev/properties/property_info
  • 解析Kernel Device Tree,设置到ro.boot.{}中
  • 解析Kernel Command Line,设置到ro.boot.{}中
  • 解析Boot Config,设置到ro.boot.{}中
  • 将一些Kernel Boot的属性,设置出去。首先获取src_prop的属性值,然后设置到dst_prop的属性中
  • 获取和设置各分区的属性,以及初始化必要的属性,这个是设置属性最多,修改也最频繁的部分。
    • 它会读取各分区的build.prop或default.prop文件获取里面的属性
    • 从init第二阶段的资源中获取属性
    • 从ramdisk中获取debug属性
    • 从厂商自定的属性文件中获取属性
    • 根据各分区属性,设置动态的只读属性(比如fingerprint)
    • 初始化build id属性
    • ……

将一些Kernel Boot的属性,设置出去

C++

    struct {
        const char* src_prop;
        const char* dst_prop;
        const char* default_value;
    } prop_map[] = {
            // clang-format off
        { "ro.boot.serialno",   "ro.serialno",   UNSET, },
        { "ro.boot.mode",       "ro.bootmode",   "unknown", },
        { "ro.boot.baseband",   "ro.baseband",   "unknown", },
        { "ro.boot.bootloader", "ro.bootloader", "unknown", },
        { "ro.boot.hardware",   "ro.hardware",   "unknown", },
        { "ro.boot.revision",   "ro.revision",   "0", },
            // clang-format on
    };

显示解析Kernel Device Tree、Boot Config和Kernel Command Line得到的属性

undefined

[ro.boot.avb_version]: [1.1]
[ro.boot.baseband]: [mdm]
[ro.boot.boot_devices]: [soc/1d84000.ufshc]
[ro.boot.bootdevice]: [1d84000.ufshc]
[ro.boot.bootreason]: [reboot,rescueparty]
[ro.boot.cert]: [M2105K81C]
[ro.boot.console]: [ttyMSM0]
[ro.boot.cpuid]: [0x5290083c]
[ro.boot.dp]: [0xB]
[ro.boot.dtb_idx]: [0]
[ro.boot.dtbo_idx]: [19]
[ro.boot.dynamic_partitions]: [true]
[ro.boot.flash.locked]: [0]
[ro.boot.force_normal_boot]: [1]
[ro.boot.fpsensor]: [goodix]
[ro.boot.fstab_suffix]: [default]
[ro.boot.hardware]: [qcom]
[ro.boot.hwc]: [CN]
[ro.boot.hwlevel]: [P2]
[ro.boot.hwversion]: [B.2.0]
[ro.boot.keymaster]: [1]
[ro.boot.memcg]: [1]
[ro.boot.oled_wp]: [808374]
[ro.boot.ramdump]: [disable]
[ro.boot.secureboot]: [1]
[ro.boot.serialno]: [9b12f90d]
[ro.boot.slot_suffix]: [_a]
[ro.boot.usbcontroller]: [a600000.dwc3]
[ro.boot.vbmeta.avb_version]: [1.0]
[ro.boot.vbmeta.device_state]: [unlocked]
[ro.boot.vbmeta.digest]: [284ee39262f0f591db3532b0ba3b514b6ba7a92042ca8f963c07dc26ca5f7303]
[ro.boot.vbmeta.hash_alg]: [sha256]
[ro.boot.vbmeta.invalidate_on_error]: [yes]
[ro.boot.vbmeta.size]: [7808]
[ro.boot.verifiedbootstate]: [orange]
[ro.boot.veritymode]: [enforcing]

挂载额外的分区,即apex和linkerconfig分区,他们都是内存分区

初始化selinux label

C++

// selinux_android_file_context_handle() takes on the order of 10+ms to run, so we want to cache
// its value.  selinux_android_restorecon() also needs an sehandle for file context look up.  It
// will create and store its own copy, but selinux_android_set_sehandle() can be used to provide
// one, thus eliminating an extra call to selinux_android_file_context_handle().
void SelabelInitialize() {
    sehandle = selinux_android_file_context_handle();
    selinux_android_set_sehandle(sehandle);
}

读取文件中记录的selinux label

C++

struct selabel_handle* selinux_android_file_context_handle(void)
{
    struct selinux_opt seopts_file[MAX_FILE_CONTEXT_SIZE];
    int size = 0;
    unsigned int i;
    for (i = 0; i < ARRAY_SIZE(seopts_file_plat); i++) {
        if (access(seopts_file_plat[i].value, R_OK) != -1) {
            seopts_file[size++] = seopts_file_plat[i];
            break;
        }
    }
    for (i = 0; i < ARRAY_SIZE(seopts_file_apex); i++) {
        if (access(seopts_file_apex[i].value, R_OK) != -1) {
            seopts_file[size++] = seopts_file_apex[i];
            break;
        }
    }
    for (i = 0; i < ARRAY_SIZE(seopts_file_system_ext); i++) {
        if (access(seopts_file_system_ext[i].value, R_OK) != -1) {
            seopts_file[size++] = seopts_file_system_ext[i];
            break;
        }
    }
    for (i = 0; i < ARRAY_SIZE(seopts_file_product); i++) {
        if (access(seopts_file_product[i].value, R_OK) != -1) {
            seopts_file[size++] = seopts_file_product[i];
            break;
        }
    }
    for (i = 0; i < ARRAY_SIZE(seopts_file_vendor); i++) {
        if (access(seopts_file_vendor[i].value, R_OK) != -1) {
            seopts_file[size++] = seopts_file_vendor[i];
            break;
        }
    }
    for (i = 0; i < ARRAY_SIZE(seopts_file_odm); i++) {
        if (access(seopts_file_odm[i].value, R_OK) != -1) {
            seopts_file[size++] = seopts_file_odm[i];
            break;
        }
    }
    return selinux_android_file_context(seopts_file, size);
}

其中find_existing_files是

C++

static const struct selinux_opt seopts_file_plat[] = {
    { SELABEL_OPT_PATH, "/system/etc/selinux/plat_file_contexts" },
    { SELABEL_OPT_PATH, "/plat_file_contexts" }
};
static const struct selinux_opt seopts_file_apex[] = {
    { SELABEL_OPT_PATH, "/dev/selinux/apex_file_contexts" }
};
static const struct selinux_opt seopts_file_system_ext[] = {
    { SELABEL_OPT_PATH, "/system_ext/etc/selinux/system_ext_file_contexts" },
    { SELABEL_OPT_PATH, "/system_ext_file_contexts" }
};
static const struct selinux_opt seopts_file_product[] = {
    { SELABEL_OPT_PATH, "/product/etc/selinux/product_file_contexts" },
    { SELABEL_OPT_PATH, "/product_file_contexts" }
};
static const struct selinux_opt seopts_file_vendor[] = {
    { SELABEL_OPT_PATH, "/vendor/etc/selinux/vendor_file_contexts" },
    { SELABEL_OPT_PATH, "/vendor_file_contexts" }
};
static const struct selinux_opt seopts_file_odm[] = {
    { SELABEL_OPT_PATH, "/odm/etc/selinux/odm_file_contexts" },
    { SELABEL_OPT_PATH, "/odm_file_contexts" }
};

从名字上,就可以看出,它是处理Init下一步行动的管理者。 它具有一个事件队列event_queue_行动队列actions_,以及一个当前正在执行的行动队列current_executing_actions_

下面分别介绍ActionManager具备的一些方法

C

void ActionManager::QueueBuiltinAction(BuiltinFunction func, const std::string& name) {
    auto lock = std::lock_guard{event_queue_lock_};
    auto action = std::make_unique<Action>(true, nullptr, "<Builtin Action>", 0, name,
                                           std::map<std::string, std::string>{});
    action->AddCommand(std::move(func), {name}, 0);

    event_queue_.emplace(action.get());
    actions_.emplace_back(std::move(action));
}

它用于构建一些额外的指令,以下面这个指令为例子

C

am.QueueBuiltinAction(SetupCgroupsAction, "SetupCgroups");
am.QueueBuiltinAction(SetKptrRestrictAction, "SetKptrRestrict");
am.QueueBuiltinAction(TestPerfEventSelinuxAction, "TestPerfEventSelinux");

它的作用是往事件队列中添加一个事件(Event)和其对应的行动(Action)。

am.QueueBuiltinAction(SetupCgroupsAction, “SetupCgroups”);

比如上面这一条指令,它的作用是将一个SetupCgroups事件和其对应的Action分别填装到事件队列行动队列中。

执行队列中的一条指令。他会检查正在执行的行动队列是否还有需要执行的指令,如果没有指令,则会检查事件队列是否还有需要执行的Event。如果当前有Event需要执行,那么它接下来会去遍历行动队列,找到下一个事件需要执行的Action,并且将它们添加到正在执行的行动队列中。 当遍历结束后,会将这个Event出队列

CheckEvent这里,会通过CheckPropertyTriggers判断当前属性是否满足执行某些Action

C

    {
        auto lock = std::lock_guard{event_queue_lock_};
        // Loop through the event queue until we have an action to execute
        while (current_executing_actions_.empty() && !event_queue_.empty()) {
            for (const auto& action : actions_) {
                if (std::visit([&action](const auto& event) { return action->CheckEvent(event); },
                               event_queue_.front())) {
                    current_executing_actions_.emplace(action.get());
                }
            }
            event_queue_.pop();
        }
    }

最后会不断执行正在执行的行动队列中的Action,每次调用一次ExecuteOneCommand(Init会循环执行这个操作),都会执行当前Action中的一条指令,通过current_command_来判断当前执行到Action的第几条指令,如果一个Action中的指令都执行完毕,那么这个Action会从正在执行的行动队列中弹出,直到所有的Action都执行完毕。然后会根据下一个Event来填充之后需要执行的Action。

C

    if (current_executing_actions_.empty()) {
        return;
    }

    auto action = current_executing_actions_.front();

    if (current_command_ == 0) {
        std::string trigger_name = action->BuildTriggersString();
        LOG(INFO) << "processing action (" << trigger_name << ") from (" << action->filename()
                  << ":" << action->line() << ")";
    }

    action->ExecuteOneCommand(current_command_);

    // If this was the last command in the current action, then remove
    // the action from the executing list.
    // If this action was oneshot, then also remove it from actions_.
    ++current_command_;
    if (current_command_ == action->NumCommands()) {
        current_executing_actions_.pop();
        current_command_ = 0;
        if (action->oneshot()) {
            auto eraser = [&action](std::unique_ptr<Action>& a) { return a.get() == action; };
            actions_.erase(std::remove_if(actions_.begin(), actions_.end(), eraser),
                           actions_.end());
        }
    }

既然这个方法与是否满足某个属性值有关,那么我们来看看它的逻辑吧。 在调用CheckEvent的时候,有以下三种情况。

  • 事件触发 - 此时Event的内容是一个String字符串
  • 属性触发 - 此时Event的内容是一个Pair对,保存着变化属性的key、value
  • 构建行为触发 - 此时Event的内容就是一个Action std::visit的作用就是为了让第二个参数自动匹配重载方法(第一个参数是具有多个重载方法的函数名)

C

bool Action::CheckEvent(const EventTrigger& event_trigger) const {
    return event_trigger == event_trigger_ && CheckPropertyTriggers();
}

bool Action::CheckEvent(const PropertyChange& property_change) const {
    const auto& [name, value] = property_change;
    return event_trigger_.empty() && CheckPropertyTriggers(name, value);
}

bool Action::CheckEvent(const BuiltinAction& builtin_action) const {
    return this == builtin_action;
}

之后让我们看看它的属性判断方式,其实就是判断传入的属性键值是否存在而已。

C

// This function checks that all property triggers are satisfied, that is
// for each (name, value) in property_triggers_, check that the current
// value of the property 'name' == value.
//
// It takes an optional (name, value) pair, which if provided must
// be present in property_triggers_; it skips the check of the current
// property value for this pair.
bool Action::CheckPropertyTriggers(const std::string& name = "", const std::string& value = "") const {
    if (property_triggers_.empty()) {
        return true;
    }

    if (!name.empty()) {
        auto it = property_triggers_.find(name);
        if (it == property_triggers_.end()) {
            return false;
        }
        const auto& trigger_value = it->second;
        if (trigger_value != "*" && trigger_value != value) {
            return false;
        }
    }

    for (const auto& [trigger_name, trigger_value] : property_triggers_) {
        if (trigger_name != name) {
            std::string prop_value = android::base::GetProperty(trigger_name, "");
            if (trigger_value == "*" && !prop_value.empty()) {
                continue;
            }
            if (trigger_value != prop_value) return false;
        }
    }
    return true;
}

如果值使用了*,并且属性值不为空,则返回true,即该属性存在即可触发。

例如:

C

on property:ro.build.ab_update=*
    mkdir /mnt/rescue/mqsas 0775 root system

上面这个只要ro.build.ab_update属性存在,就会执行创建mqsas文件夹的逻辑。

它是属性服务,最重要的职责就是处理设置属性的事件。我们在深入了解PropertyService之后,就能明白属性有着各种各样的类别与作用,有的属性是控制属性,有的属性是动态生成的。 当然,PropertyService是由init通过执行StartPropertyService启动的。

我们不会花大量篇幅来讲述PropertyService是如何工作的,只会讲述其中一些重要的方法

调用链:StartPropertyService -> [PropertyServiceThread](此处会堵塞等待消息) -> handle_property_set_fd -> HandlePropertySet -> PropertySet -> PropertyChanged

Init中存在一个PropertyChanged方法,当PropertyService进行属性修改时,会通过PropertyChanged方法通知init,当前有属性发生了变更。

C

void PropertyChanged(const std::string& name, const std::string& value) {
    // If the property is sys.powerctl, we bypass the event queue and immediately handle it.
    // This is to ensure that init will always and immediately shutdown/reboot, regardless of
    // if there are other pending events to process or if init is waiting on an exec service or
    // waiting on a property.
    // In non-thermal-shutdown case, 'shutdown' trigger will be fired to let device specific
    // commands to be executed.
    if (name == "sys.powerctl") {
        trigger_shutdown(value);
    }

    if (property_triggers_enabled) {
        ActionManager::GetInstance().QueuePropertyChange(name, value);
        WakeMainInitThread();
    }

    prop_waiter_state.CheckAndResetWait(name, value);
}

PropertyChanged做的事情非常简单,只做三个事情

  • 当接收到传递了sys.powerctl属性时,会调用ShutdownState来执行相关的事件,一般是处理重启、关机等操作
  • 当Init开始启用属性变更检查时(时机是在late-init之后,由property_triggers_enabled变量进行控制),会将变化的属性键值添加到事件队列中,唤醒init进行处理
  • 如果此前设置了需要等待某个属性,当前改变的属性正好匹配,就会唤醒init进行处理

这个方法是Init中最重要的方法之一,它的作用是从init的脚本文件中加载事件和对应的行动。

C

static void LoadBootScripts(ActionManager& action_manager, ServiceList& service_list) {
    Parser parser = CreateParser(action_manager, service_list);

    std::string bootscript = GetProperty("ro.boot.init_rc", "");
    if (bootscript.empty()) {
        parser.ParseConfig("/system/etc/init/hw/init.rc");
        if (!parser.ParseConfig("/system/etc/init")) {
            late_import_paths.emplace_back("/system/etc/init");
        }
        // late_import is available only in Q and earlier release. As we don't
        // have system_ext in those versions, skip late_import for system_ext.
        parser.ParseConfig("/system_ext/etc/init");
        if (!parser.ParseConfig("/vendor/etc/init")) {
            late_import_paths.emplace_back("/vendor/etc/init");
        }
        if (!parser.ParseConfig("/odm/etc/init")) {
            late_import_paths.emplace_back("/odm/etc/init");
        }
        if (!parser.ParseConfig("/product/etc/init")) {
            late_import_paths.emplace_back("/product/etc/init");
        }
    } else {
        parser.ParseConfig(bootscript);
    }
}

Parser在通过ParseConfig解析路径时,会通过判断该路径是文件还是文件夹。

  • 如果是文件,那么Parser会直接解析这个文件
  • 如果是文件夹,那么Parser会遍历这个文件夹(子文件夹不会被遍历),将所有的文件加入到解析列表中

接下来,我们分析一下解析文件的方法

C

bool Parser::ParseConfigFile(const std::string& path) {
    LOG(INFO) << "Parsing file " << path << "...";
    android::base::Timer t;
    auto config_contents = ReadFile(path);
    if (!config_contents.ok()) {
        LOG(INFO) << "Unable to read config file '" << path << "': " << config_contents.error();
        return false;
    }

    ParseData(path, &config_contents.value());

    LOG(VERBOSE) << "(Parsing " << path << " took " << t << ".)";
    return true;
}

首先,他会读取脚本文件中所有的数据,转换成string。 然后通过ParseData进行解析

  • 通过ParseSection解析新的Action段落。在解析的时候,会将需要判断的属性都解析出来,形成一个属性向量,和Event放在一个Action中
  • 通过ParseLineSection解析当前Action需要的指令,然后将它添加到当前的Action中(通过ParseSection解析出来的Action)
  • 最后通过EndSection将封装好的Action放入ActionManager中的行动队列中

总而言之,init解析脚本,是按照顺序进行解析和入队列的。

举个例子

C


on late-init
    setprop sys.ota.type recovery

on late-init && property:ro.build.ab_update=true
    setprop sys.ota.type update_engine

这里最终会生成两个Action

  1. Action1 (event_trigger : late-init)
  2. Action2 (event_trigger : late-init, property_triggers: {ro.build.ab_update = true}) 由于Action1会先进入行动队列,因此后续这两个Action在都满足条件的情况下,init循环中会先执行Action1,然后再执行Action2。 init脚本指令的顺序,会影响到init执行指令的顺序。也就是说,同时满足触发条件的执行指令,它会按照从上到下的顺序执行

原文链接:https://ovea-y.cn/init_work_1/

相关内容