Add ulimit builtin and optimize shed -c to exec single commands directly without forking
This commit is contained in:
@@ -24,13 +24,14 @@ pub mod test; // [[ ]] thing
|
||||
pub mod trap;
|
||||
pub mod varcmds;
|
||||
pub mod zoltraak;
|
||||
pub mod resource;
|
||||
|
||||
pub const BUILTINS: [&str; 47] = [
|
||||
pub const BUILTINS: [&str; 49] = [
|
||||
"echo", "cd", "read", "export", "local", "pwd", "source", "shift", "jobs", "fg", "bg", "disown",
|
||||
"alias", "unalias", "return", "break", "continue", "exit", "zoltraak", "shopt", "builtin",
|
||||
"command", "trap", "pushd", "popd", "dirs", "exec", "eval", "true", "false", ":", "readonly",
|
||||
"unset", "complete", "compgen", "map", "pop", "fpop", "push", "fpush", "rotate", "wait", "type",
|
||||
"getopts", "keymap", "read_key", "autocmd",
|
||||
"getopts", "keymap", "read_key", "autocmd", "ulimit", "umask"
|
||||
];
|
||||
|
||||
pub fn true_builtin() -> ShResult<()> {
|
||||
|
||||
169
src/builtin/resource.rs
Normal file
169
src/builtin/resource.rs
Normal file
@@ -0,0 +1,169 @@
|
||||
use ariadne::Fmt;
|
||||
use nix::sys::resource::{Resource, getrlimit, setrlimit};
|
||||
use yansi::Color;
|
||||
|
||||
use crate::{
|
||||
getopt::{Opt, OptSpec, get_opts_from_tokens}, libsh::error::{ShErr, ShErrKind, ShResult, ShResultExt, next_color}, parse::{NdRule, Node, execute::prepare_argv}, prelude::*, state::{self}
|
||||
};
|
||||
|
||||
fn ulimit_opt_spec() -> [OptSpec;5] {
|
||||
[
|
||||
OptSpec {
|
||||
opt: Opt::Short('n'), // file descriptors
|
||||
takes_arg: true,
|
||||
},
|
||||
OptSpec {
|
||||
opt: Opt::Short('u'), // max user processes
|
||||
takes_arg: true,
|
||||
},
|
||||
OptSpec {
|
||||
opt: Opt::Short('s'), // stack size
|
||||
takes_arg: true,
|
||||
},
|
||||
OptSpec {
|
||||
opt: Opt::Short('c'), // core dump file size
|
||||
takes_arg: true,
|
||||
},
|
||||
OptSpec {
|
||||
opt: Opt::Short('v'), // virtual memory
|
||||
takes_arg: true,
|
||||
}
|
||||
]
|
||||
}
|
||||
|
||||
struct UlimitOpts {
|
||||
fds: Option<u64>,
|
||||
procs: Option<u64>,
|
||||
stack: Option<u64>,
|
||||
core: Option<u64>,
|
||||
vmem: Option<u64>,
|
||||
}
|
||||
|
||||
fn get_ulimit_opts(opt: &[Opt]) -> ShResult<UlimitOpts> {
|
||||
let mut opts = UlimitOpts {
|
||||
fds: None,
|
||||
procs: None,
|
||||
stack: None,
|
||||
core: None,
|
||||
vmem: None,
|
||||
};
|
||||
|
||||
for o in opt {
|
||||
match o {
|
||||
Opt::ShortWithArg('n', arg) => {
|
||||
opts.fds = Some(arg.parse().map_err(|_| ShErr::simple(
|
||||
ShErrKind::ParseErr,
|
||||
format!("invalid argument for -n: {}", arg.fg(next_color())),
|
||||
))?);
|
||||
},
|
||||
Opt::ShortWithArg('u', arg) => {
|
||||
opts.procs = Some(arg.parse().map_err(|_| ShErr::simple(
|
||||
ShErrKind::ParseErr,
|
||||
format!("invalid argument for -u: {}", arg.fg(next_color())),
|
||||
))?);
|
||||
},
|
||||
Opt::ShortWithArg('s', arg) => {
|
||||
opts.stack = Some(arg.parse().map_err(|_| ShErr::simple(
|
||||
ShErrKind::ParseErr,
|
||||
format!("invalid argument for -s: {}", arg.fg(next_color())),
|
||||
))?);
|
||||
},
|
||||
Opt::ShortWithArg('c', arg) => {
|
||||
opts.core = Some(arg.parse().map_err(|_| ShErr::simple(
|
||||
ShErrKind::ParseErr,
|
||||
format!("invalid argument for -c: {}", arg.fg(next_color())),
|
||||
))?);
|
||||
},
|
||||
Opt::ShortWithArg('v', arg) => {
|
||||
opts.vmem = Some(arg.parse().map_err(|_| ShErr::simple(
|
||||
ShErrKind::ParseErr,
|
||||
format!("invalid argument for -v: {}", arg.fg(next_color())),
|
||||
))?);
|
||||
},
|
||||
o => return Err(ShErr::simple(
|
||||
ShErrKind::ParseErr,
|
||||
format!("invalid option: {}", o.fg(next_color())),
|
||||
)),
|
||||
}
|
||||
}
|
||||
|
||||
Ok(opts)
|
||||
}
|
||||
|
||||
pub fn ulimit(node: Node) -> ShResult<()> {
|
||||
let span = node.get_span();
|
||||
let NdRule::Command {
|
||||
assignments: _,
|
||||
argv,
|
||||
} = node.class
|
||||
else {
|
||||
unreachable!()
|
||||
};
|
||||
|
||||
let (_, opts) = get_opts_from_tokens(argv, &ulimit_opt_spec()).promote_err(span.clone())?;
|
||||
let ulimit_opts = get_ulimit_opts(&opts).promote_err(span.clone())?;
|
||||
|
||||
if let Some(fds) = ulimit_opts.fds {
|
||||
let (_, hard) = getrlimit(Resource::RLIMIT_NOFILE).map_err(|e| ShErr::at(
|
||||
ShErrKind::ExecFail,
|
||||
span.clone(),
|
||||
format!("failed to get file descriptor limit: {}", e),
|
||||
))?;
|
||||
setrlimit(Resource::RLIMIT_NOFILE, fds, hard).map_err(|e| ShErr::at(
|
||||
ShErrKind::ExecFail,
|
||||
span.clone(),
|
||||
format!("failed to set file descriptor limit: {}", e),
|
||||
))?;
|
||||
}
|
||||
if let Some(procs) = ulimit_opts.procs {
|
||||
let (_, hard) = getrlimit(Resource::RLIMIT_NPROC).map_err(|e| ShErr::at(
|
||||
ShErrKind::ExecFail,
|
||||
span.clone(),
|
||||
format!("failed to get process limit: {}", e),
|
||||
))?;
|
||||
setrlimit(Resource::RLIMIT_NPROC, procs, hard).map_err(|e| ShErr::at(
|
||||
ShErrKind::ExecFail,
|
||||
span.clone(),
|
||||
format!("failed to set process limit: {}", e),
|
||||
))?;
|
||||
}
|
||||
if let Some(stack) = ulimit_opts.stack {
|
||||
let (_, hard) = getrlimit(Resource::RLIMIT_STACK).map_err(|e| ShErr::at(
|
||||
ShErrKind::ExecFail,
|
||||
span.clone(),
|
||||
format!("failed to get stack size limit: {}", e),
|
||||
))?;
|
||||
setrlimit(Resource::RLIMIT_STACK, stack, hard).map_err(|e| ShErr::at(
|
||||
ShErrKind::ExecFail,
|
||||
span.clone(),
|
||||
format!("failed to set stack size limit: {}", e),
|
||||
))?;
|
||||
}
|
||||
if let Some(core) = ulimit_opts.core {
|
||||
let (_, hard) = getrlimit(Resource::RLIMIT_CORE).map_err(|e| ShErr::at(
|
||||
ShErrKind::ExecFail,
|
||||
span.clone(),
|
||||
format!("failed to get core dump size limit: {}", e),
|
||||
))?;
|
||||
setrlimit(Resource::RLIMIT_CORE, core, hard).map_err(|e| ShErr::at(
|
||||
ShErrKind::ExecFail,
|
||||
span.clone(),
|
||||
format!("failed to set core dump size limit: {}", e),
|
||||
))?;
|
||||
}
|
||||
if let Some(vmem) = ulimit_opts.vmem {
|
||||
let (_, hard) = getrlimit(Resource::RLIMIT_AS).map_err(|e| ShErr::at(
|
||||
ShErrKind::ExecFail,
|
||||
span.clone(),
|
||||
format!("failed to get virtual memory limit: {}", e),
|
||||
))?;
|
||||
setrlimit(Resource::RLIMIT_AS, vmem, hard).map_err(|e| ShErr::at(
|
||||
ShErrKind::ExecFail,
|
||||
span.clone(),
|
||||
format!("failed to set virtual memory limit: {}", e),
|
||||
))?;
|
||||
}
|
||||
|
||||
state::set_status(0);
|
||||
Ok(())
|
||||
}
|
||||
Reference in New Issue
Block a user