Files
cleonos/cleonos/c/apps/kill_main.c

183 lines
4.3 KiB
C
Raw Normal View History

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;
}