从cgroup_init_early函数学习cgroup——初始化代码

本文详细介绍了Linux内核中的cgroup初始化过程,特别是`cgroup_init_early`函数的执行步骤。通过对`init_css_set`的初始化、`cgroup_dummy_root`的设置以及内置子系统的初始化,揭示了cgroup机制如何在系统启动早期发挥作用。文章还探讨了子系统分类、`cgroup_subsys`数组及其初始化细节。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前面分析了cgroup机制的框架及数据结构之间的关系,现在看代码就比较见简单了。
代码部分
在init/main.c的start_kernel函数中会调用cgroup_init_early函数进行cgroup的初始化。其实对cgroup的初始化分两次,分别是cgroup_init_early和cgroup_init。这么做的原因是,系统初始阶段需要使用一些 subsystem,先对这一部分进行初始化。
来看一下cgroup_init_early

/**
 * cgroup_init_early - cgroup initialization at system boot
 *
 * Initialize cgroups at system boot, and initialize any
 * subsystems that request early init.
 */
int __init cgroup_init_early(void)
{
    struct cgroup_subsys *ss;
    int i;

    atomic_set(&init_css_set.refcount, 1);
    INIT_LIST_HEAD(&init_css_set.cgrp_links);
    INIT_LIST_HEAD(&init_css_set.tasks);
    INIT_HLIST_NODE(&init_css_set.hlist);
    css_set_count = 1;
    init_cgroup_root(&cgroup_dummy_root);
    cgroup_root_count = 1;
    RCU_INIT_POINTER(init_task.cgroups, &init_css_set);

    init_cgrp_cset_link.cset = &init_css_set;
    init_cgrp_cset_link.cgrp = cgroup_dummy_top;
    list_add(&init_cgrp_cset_link.cset_link, &cgroup_dummy_top->cset_links);
    list_add(&init_cgrp_cset_link.cgrp_link, &init_css_set.cgrp_links);

    /* at bootup time, we don't worry about modular subsystems */
    for_each_builtin_subsys(ss, i) {
        BUG_ON(!ss->name);
        BUG_ON(strlen(ss->name) > MAX_CGROUP_TYPE_NAMELEN);
        BUG_ON(!ss->css_alloc);
        BUG_ON(!ss->css_free);
        if (ss->subsys_id != i) {
            printk(KERN_ERR "cgroup: Subsys %s id == %d\n",
                   ss->name, ss->subsys_id);
            BUG();
        }

        if (ss->early_init)
            cgroup_init_subsys(ss);
    }
    return 0;
}

注释的说的很清楚了,initialize any subsystems that request early init.
init_css_set是init进程使用的css_set结构体,先进行初始化:
设置init_css_set的引用计数为1.
初始化init_css_set.cgrp_links链表,该链表与cgrp_cset_link的cgrp_link相关联.
初始化init_css_set.tasks链表,该链表与所有应用init_css_set的task相关联.
初始化init_css_set.hlist链表,哈希表相关,没有深入研究.
css_set_count用来记录系统css_set结构体变量的个数.
初始化cgroup_dummy_root,定义为

static struct cgroupfs_root cgroup_dummy_root;
/*
 * A cgroupfs_root represents the root of a cgroup hierarchy, and may be
 * associated with a superblock to form an active hierarchy.  This is
 * internal to cgroup core.  Don't access directly from controllers.
 */
struct cgroupfs_root {
    struct super_block *sb;

    /* The bitmask of subsystems attached to this hierarchy */
    unsigned long subsys_mask;

    /* Unique id for this hier
<think>好的,用户想了解如何正确配置cgroup_enable=cpuset在Linux系统中。首先,我需要回忆一下cgroup的基本概念,cgroups是Linux内核的一个功能,用来限制、控制和统计进程组的资源使用。cpuset子系统用于将进程绑定到特定的CPU和内存节点。 用户提到了cgroup_enable=cpuset,这可能是在启动参数中添加的选项。根据引用[1],用户之前遇到问题需要添加“cgroup_memory=1 cgroup_enable=memory”到linux cmdline,所以类似的,cgroup_enable=cpuset可能也需要在启动参数中配置。 接下来,我需要确定如何修改Linux的启动参数。不同的发行版有不同的方法,比如在Debian/Ubuntu系统中,通常修改/boot/cmdline.txt或者grub配置文件。而在Raspberry Pi的例子中,引用[1]提到是在/boot/cmdline.txt中添加参数,所以这可能是一个常见的位置。 步骤可能包括:编辑grub配置文件,更新grub,然后重启。同时,需要确保cgroup相关的工具已经安装,比如libcgroup-tools,如引用[3]中提到的安装步骤。此外,配置资源组可能需要创建cgroup的层级结构,并设置相应的参数。 还需要验证配置是否生效,可以通过检查/proc/cmdline或者查看cgroup的文件系统。另外,用户的问题可能涉及到其他子系统,比如memory,所以需要确认是否有其他相关参数需要同时启用。 可能的相关问题包括如何检查cgroup是否正确加载,或者配置后资源限制没有生效怎么办。需要确保用户不仅修改了启动参数,还正确设置了cpuset的具体配置,比如分配CPU核心和内存节点。 总结步骤应该是:1. 修改启动参数添加cgroup_enable=cpuset;2. 安装必要工具;3. 配置cpuset子系统;4. 验证配置。同时要注意不同发行版的差异,比如使用systemd的系统可能需要不同的方法。</think>### 配置 `cgroup_enable=cpuset` 的步骤 #### 1. **修改Linux启动参数** - **对于Raspberry Pi等设备**: 编辑 `/boot/cmdline.txt` 文件,在现有参数末尾添加 `cgroup_enable=cpuset`,例如: ```bash console=serial0,115200 console=tty1 root=PARTUUID=58f8c84c-02 rootfstype=ext4 cgroup_enable=cpuset cgroup_memory=1 fsck.repair=yes rootwait ``` 保存后重启系统[^1]。 - **对于通用Linux系统(如Ubuntu/CentOS)**: 修改GRUB配置文件(通常为 `/etc/default/grub`),在 `GRUB_CMDLINE_LINUX` 行追加参数: ```bash GRUB_CMDLINE_LINUX="... cgroup_enable=cpuset" ``` 更新GRUB并重启: ```bash sudo update-grub && sudo reboot ``` #### 2. **验证配置是否生效** - 检查内核启动参数: ```bash cat /proc/cmdline | grep cgroup_enable=cpuset ``` - 确认cpuset子系统已挂载: ```bash mount | grep cgroup | grep cpuset ``` #### 3. **配置cpuset子系统(可选)** - 创建自定义cgroup层级结构(需安装 `libcgroup-tools`): ```bash sudo cgcreate -g cpuset:/my_group ``` - 分配CPU核心和内存节点: ```bash echo 0-1 > /sys/fs/cgroup/cpuset/my_group/cpuset.cpus echo 0 > /sys/fs/cgroup/cpuset/my_group/cpuset.mems ``` #### 4. **关联进程到cgroup** ```bash sudo cgexec -g cpuset:/my_group your_command ``` --- ### 关键问题解析 1. **为何需要同时启用 `cgroup_memory=1`?** cpuset子系统依赖内存控制功能,若未启用可能导致资源分配异常[^1][^3]。 2. **配置后未生效的排查方法** - 检查内核日志:`dmesg | grep cgroup` - 确认cgroup文件系统挂载点:`ls /sys/fs/cgroup/cpuset` ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值