Skip to content

17.3 资源限制

FreeBSD 提供 /etc/login.conf 静态资源限制与 rctl 动态资源限制两种机制。本节分别给出登录类别(login class)配置方法和进程级约束规则定义。

17.3.1 资源限制

在 FreeBSD 中,资源限制是指对进程和用户可使用的系统资源进行控制与管理的机制,用于防止单个进程或用户消耗过多资源,避免性能下降或系统不稳定。资源限制有助于确保系统中所有活动进程和用户间的资源公平分配。

FreeBSD 提供多种方法供管理员限制用户可使用的系统资源量。

传统方法是通过编辑 /etc/login.conf 文件来定义登录类别。虽然该方法仍受支持,但任何更改都需经过多个步骤:编辑该文件、重建资源数据库、修改 /etc/master.passwd 文件,以及重建密码数据库。这一过程较为繁琐,耗时取决于需配置的用户数量。

与之相比,rctl 提供了一种更细粒度的资源限制机制,它还可为进程和 jail 设置资源约束。

本节演示两种控制资源的方法,从传统方法开始。

17.3.1.1 资源类型

FreeBSD 支持对以下资源类型施加限制:

类型描述
cputime限制进程可消耗的 CPU 时间量
memoryuse / memorylocked控制进程可使用的物理内存量 / 锁定内存量
openfiles限制进程可同时打开的文件数量
maxproc控制用户或进程可创建的进程数量
filesize限制进程可创建的文件的最大大小
coredumpsize控制是否允许进程生成核心转储文件及其大小
datasize / stacksize限制数据段 / 栈段大小
vmemoryuse限制进程的总虚拟存储器使用量
sbsize限制套接字缓冲区大小
pseudoterminals限制伪终端数量
swapuse限制交换空间使用量
nthr限制线程数(仅 rctl
wallclock / pcpu限制墙钟时间 / CPU 百分比(仅 rctl
readbps / writebps / readiops / writeiops限制文件系统读写速率(仅 rctl

资源类型表

17.3.1.2 配置登录类别

传统方法中,登录类别及其关联的资源限制定义于 /etc/login.conf 文件。每个用户账户可归入某个登录类别,default 为默认类别。各类别关联一组登录能力(login capability),格式为 名称=值 对,其中 名称 是已知标识符, 为任意字符串,按对应 名称 的规则处理。

配置资源限制需先编辑 /etc/login.conf 文件,定位至要修改的用户类别节。

本例假设所用类别名为 limited,如果不存在则创建。

ini
limited:\	# 用户类别名称
        :maxproc=50:\ # 将 limited 类别下用户的最大进程数(maxproc)设为 50
        :tc=default: # 此用户类别继承 default 类别的默认设置

修改 /etc/login.conf 后,运行 cap_mkdb 生成数据库,供 FreeBSD 应用这些设置:

sh
# cap_mkdb /etc/login.conf

chpass 可为目标用户 ykla 切换类别:

sh
# chpass ykla

技巧

使用命令 export EDITOR=ee 可将默认编辑器由 vi 变更为 ee,便于操作。

这将打开文本编辑器,按如下方式添加新的 limited 类别:

ini
#Changing user information for ykla.
Login: ykla
Password: $6$SqMJXrv5aC6Wq.by$nmbZs078aHNBVyh9noLFouJsGHyFSvQIzH0W4zpdfXuPtGtt.FHgWfXDHVBa
Uid [#]: 1001
Gid [# or name]: 1001
Change [month day year]:
Expire [month day year]: April 16, 2027
Class: limited
Home directory: /home/ykla
Shell: /bin/sh
Full Name: User &
Office Location:
Office Phone:
Home Phone:
Other information:

现归属于 limited 类别的用户,其最大进程数将限制为 50。注意,以上仅为使用 /etc/login.conf 设置资源限制的一个示例。

修改 /etc/login.conf 后,用户需注销并重新登录才能使更改生效。

17.3.2 启用和配置资源限制

rctl 命令用于控制资源限制,可动态分配资源约束至特定进程或用户,与所属用户类别无关。

使用 rctl 前需先将其启用,在 /boot/loader.conf 中添加以下行并重启系统:

ini
kern.racct.enable=1

再执行以下命令启用并启动 rctl 服务:

sh
# service rctl enable
# service rctl start

此后即可使用 rctl 为系统设置规则。

规则语法含主语processuserloginclassjail)、主语-id、资源与动作四部分:

text
主语:主语-id:资源:动作=数量/对象

其中 对象 定义数量统计范围:

标识路径作用范围说明
/process每进程对单个进程分别应用资源限制。
/user每用户对同一用户的所有进程统一限制。
/loginclass每登录类别按登录类别(login class)分组限制。

省略时默认与主语一致。

常见可用动作:

动作说明
deny拒绝该资源分配或操作。
log不阻止操作,仅在控制台记录警告信息。
devctl向 devd(8) 发送事件通知,由系统设备守护进程处理。

例如,要约束用户 ykla 最多只能创建 10 个进程,执行:

sh
# rctl -a user:ykla:maxproc:deny=10/user

要检查已应用的资源限制,执行 rctl:

sh
# rctl

输出应类似于以下内容:

text
user:ykla:maxproc:deny=10

将规则写入 /etc/rctl.conf 文件,重启后仍可生效。格式为单条规则,无需前导命令。例如,前述规则可添加为:

ini
user:ykla:maxproc:deny=10

17.3.3 课后习题

  1. /etc/login.conf 文件中为一个测试用户类设置 maxproc 为 50,然后编写一个简单的 Shell 脚本在该类用户下创建超过 50 个子进程,观察系统行为并分析内核通过何种机制阻止超限。

  2. 使用 limits -a 查看当前 Shell 的所有资源限制,将这些限制与 sysctl kern.maxproc 的输出进行对比,指出用户级限制与系统全局上限之间的关系。

  3. 已知 fork 炸弹 :(){ :|:& };: 可能拖垮未做限制的系统。设计一个实验:如何防止这种问题。