2026-04-17 17:07:19 +08:00
|
|
|
#include "cmd_runtime.h"
|
|
|
|
|
|
2026-04-17 17:28:49 +08:00
|
|
|
static void ush_kill_upper_copy(char *dst, u64 dst_size, const char *src) {
|
|
|
|
|
u64 i = 0ULL;
|
|
|
|
|
|
|
|
|
|
if (dst == (char *)0 || dst_size == 0ULL) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dst[0] = '\0';
|
|
|
|
|
|
|
|
|
|
if (src == (const char *)0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
while (src[i] != '\0' && i + 1ULL < dst_size) {
|
|
|
|
|
char ch = src[i];
|
|
|
|
|
if (ch >= 'a' && ch <= 'z') {
|
|
|
|
|
ch = (char)(ch - ('a' - 'A'));
|
|
|
|
|
}
|
|
|
|
|
dst[i] = ch;
|
|
|
|
|
i++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dst[i] = '\0';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int ush_kill_parse_signal_name(const char *text, u64 *out_signal) {
|
|
|
|
|
char name[32];
|
|
|
|
|
u64 start = 0ULL;
|
|
|
|
|
|
|
|
|
|
if (text == (const char *)0 || out_signal == (u64 *)0 || text[0] == '\0') {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ush_kill_upper_copy(name, (u64)sizeof(name), text);
|
|
|
|
|
|
|
|
|
|
if (name[0] == '-') {
|
|
|
|
|
start = 1ULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (name[start] == 'S' && name[start + 1ULL] == 'I' && name[start + 2ULL] == 'G') {
|
|
|
|
|
start += 3ULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (name[start] == '\0') {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ush_streq(&name[start], "TERM") != 0) {
|
|
|
|
|
*out_signal = CLEONOS_SIGTERM;
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ush_streq(&name[start], "KILL") != 0) {
|
|
|
|
|
*out_signal = CLEONOS_SIGKILL;
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ush_streq(&name[start], "STOP") != 0) {
|
|
|
|
|
*out_signal = CLEONOS_SIGSTOP;
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ush_streq(&name[start], "CONT") != 0) {
|
|
|
|
|
*out_signal = CLEONOS_SIGCONT;
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int ush_kill_parse_signal(const char *text, u64 *out_signal) {
|
|
|
|
|
u64 parsed = 0ULL;
|
|
|
|
|
|
|
|
|
|
if (text == (const char *)0 || out_signal == (u64 *)0 || text[0] == '\0') {
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ush_parse_u64_dec(text, &parsed) != 0 && parsed <= 255ULL) {
|
|
|
|
|
*out_signal = parsed;
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ush_kill_parse_signal_name(text, out_signal);
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-17 17:07:19 +08:00
|
|
|
static int ush_cmd_kill(const char *arg) {
|
|
|
|
|
char pid_text[USH_PATH_MAX];
|
2026-04-17 17:28:49 +08:00
|
|
|
char signal_text[USH_PATH_MAX];
|
2026-04-17 17:07:19 +08:00
|
|
|
const char *rest = "";
|
|
|
|
|
u64 pid;
|
2026-04-17 17:28:49 +08:00
|
|
|
u64 signal = CLEONOS_SIGTERM;
|
2026-04-17 17:07:19 +08:00
|
|
|
u64 ret;
|
|
|
|
|
|
|
|
|
|
if (arg == (const char *)0 || arg[0] == '\0') {
|
2026-04-17 17:28:49 +08:00
|
|
|
ush_writeln("kill: usage kill <pid> [TERM|KILL|STOP|CONT|signal]");
|
2026-04-17 17:07:19 +08:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ush_split_first_and_rest(arg, pid_text, (u64)sizeof(pid_text), &rest) == 0) {
|
2026-04-17 17:28:49 +08:00
|
|
|
ush_writeln("kill: usage kill <pid> [TERM|KILL|STOP|CONT|signal]");
|
2026-04-17 17:07:19 +08:00
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ush_parse_u64_dec(pid_text, &pid) == 0 || pid == 0ULL) {
|
|
|
|
|
ush_writeln("kill: invalid pid");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (rest != (const char *)0 && rest[0] != '\0') {
|
2026-04-17 17:28:49 +08:00
|
|
|
ush_copy(signal_text, (u64)sizeof(signal_text), rest);
|
|
|
|
|
ush_trim_line(signal_text);
|
|
|
|
|
if (ush_kill_parse_signal(signal_text, &signal) == 0) {
|
2026-04-17 17:07:19 +08:00
|
|
|
ush_writeln("kill: invalid signal");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ret = cleonos_sys_proc_kill(pid, signal);
|
|
|
|
|
|
|
|
|
|
if (ret == (u64)-1) {
|
|
|
|
|
ush_writeln("kill: pid not found");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (ret == 0ULL) {
|
|
|
|
|
ush_writeln("kill: target cannot be terminated right now");
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ush_write("kill: sent signal ");
|
|
|
|
|
ush_write_hex_u64(signal);
|
|
|
|
|
ush_write(" to ");
|
|
|
|
|
ush_write_hex_u64(pid);
|
|
|
|
|
ush_write_char('\n');
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int cleonos_app_main(void) {
|
|
|
|
|
ush_cmd_ctx ctx;
|
|
|
|
|
ush_cmd_ret ret;
|
|
|
|
|
ush_state sh;
|
|
|
|
|
char initial_cwd[USH_PATH_MAX];
|
|
|
|
|
int has_context = 0;
|
|
|
|
|
int success = 0;
|
|
|
|
|
const char *arg = "";
|
|
|
|
|
|
|
|
|
|
ush_zero(&ctx, (u64)sizeof(ctx));
|
|
|
|
|
ush_zero(&ret, (u64)sizeof(ret));
|
|
|
|
|
ush_init_state(&sh);
|
|
|
|
|
ush_copy(initial_cwd, (u64)sizeof(initial_cwd), sh.cwd);
|
|
|
|
|
|
|
|
|
|
if (ush_command_ctx_read(&ctx) != 0) {
|
|
|
|
|
if (ctx.cmd[0] != '\0' && ush_streq(ctx.cmd, "kill") != 0) {
|
|
|
|
|
has_context = 1;
|
|
|
|
|
arg = ctx.arg;
|
|
|
|
|
if (ctx.cwd[0] == '/') {
|
|
|
|
|
ush_copy(sh.cwd, (u64)sizeof(sh.cwd), ctx.cwd);
|
|
|
|
|
ush_copy(initial_cwd, (u64)sizeof(initial_cwd), sh.cwd);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
success = ush_cmd_kill(arg);
|
|
|
|
|
|
|
|
|
|
if (has_context != 0) {
|
|
|
|
|
if (ush_streq(sh.cwd, initial_cwd) == 0) {
|
|
|
|
|
ret.flags |= USH_CMD_RET_FLAG_CWD;
|
|
|
|
|
ush_copy(ret.cwd, (u64)sizeof(ret.cwd), sh.cwd);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (sh.exit_requested != 0) {
|
|
|
|
|
ret.flags |= USH_CMD_RET_FLAG_EXIT;
|
|
|
|
|
ret.exit_code = sh.exit_code;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
(void)ush_command_ret_write(&ret);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (success != 0) ? 0 : 1;
|
|
|
|
|
}
|