Shell: command execution and parsing support piped commands

There is still problems with "blocking" builtin commands (time),
return value, ...
This commit is contained in:
Bananymous 2023-08-17 12:05:38 +03:00
parent 85c6149138
commit 089da2608c
1 changed files with 194 additions and 58 deletions

View File

@ -21,7 +21,7 @@ extern char** environ;
static const char* argv0 = nullptr;
static int last_return = 0;
BAN::Vector<BAN::String> parse_command(BAN::StringView);
BAN::Vector<BAN::Vector<BAN::String>> parse_command(BAN::StringView);
BAN::Optional<BAN::String> parse_dollar(BAN::StringView command, size_t& i)
{
@ -151,7 +151,7 @@ BAN::Optional<BAN::String> parse_dollar(BAN::StringView command, size_t& i)
return "$"sv;
}
BAN::Vector<BAN::String> parse_command(BAN::StringView command)
BAN::Vector<BAN::Vector<BAN::String>> parse_command(BAN::StringView command_view)
{
enum class State
{
@ -160,13 +160,14 @@ BAN::Vector<BAN::String> parse_command(BAN::StringView command)
DoubleQuote,
};
BAN::Vector<BAN::String> result;
BAN::Vector<BAN::Vector<BAN::String>> result;
BAN::Vector<BAN::String> command_args;
State state = State::Normal;
BAN::String current;
for (size_t i = 0; i < command.size(); i++)
BAN::String current_arg;
for (size_t i = 0; i < command_view.size(); i++)
{
char c = command[i];
char c = command_view[i];
switch (state)
{
@ -177,22 +178,31 @@ BAN::Vector<BAN::String> parse_command(BAN::StringView command)
state = State::DoubleQuote;
else if (c == '$')
{
auto expansion = parse_dollar(command, i);
auto expansion = parse_dollar(command_view, i);
if (!expansion.has_value())
{
fprintf(stderr, "bad substitution\n");
return {};
}
MUST(current.append(expansion.value()));
MUST(current_arg.append(expansion.value()));
}
else if (c == '|')
{
if (!current_arg.empty())
MUST(command_args.push_back(current_arg));
current_arg.clear();
MUST(result.push_back(command_args));
command_args.clear();
}
else if (!isspace(c))
MUST(current.push_back(c));
MUST(current_arg.push_back(c));
else
{
if (!current.empty())
if (!current_arg.empty())
{
MUST(result.push_back(current));
current.clear();
MUST(command_args.push_back(current_arg));
current_arg.clear();
}
}
break;
@ -200,43 +210,59 @@ BAN::Vector<BAN::String> parse_command(BAN::StringView command)
if (c == '\'')
state = State::Normal;
else
MUST(current.push_back(c));
MUST(current_arg.push_back(c));
break;
case State::DoubleQuote:
if (c == '"')
state = State::Normal;
else if (c != '$')
MUST(current.push_back(c));
MUST(current_arg.push_back(c));
else
{
auto expansion = parse_dollar(command, i);
auto expansion = parse_dollar(command_view, i);
if (!expansion.has_value())
{
fprintf(stderr, "bad substitution\n");
return {};
}
MUST(current.append(expansion.value()));
MUST(current_arg.append(expansion.value()));
}
break;
}
}
// FIXME: handle state != State::Normal
MUST(result.push_back(BAN::move(current)));
MUST(command_args.push_back(BAN::move(current_arg)));
MUST(result.push_back(BAN::move(command_args)));
return result;
}
int execute_command(BAN::Vector<BAN::String>& args)
int execute_command(BAN::Vector<BAN::String>& args, int fd_in, int fd_out);
BAN::Optional<int> execute_builtin(BAN::Vector<BAN::String>& args, int fd_in, int fd_out)
{
if (args.empty())
return 0;
FILE* fout = stdout;
bool should_close = false;
if (fd_out != STDOUT_FILENO)
{
int fd_dup = dup(fd_out);
if (fd_dup == -1)
ERROR_RETURN("dup", 1);
fout = fdopen(fd_dup, "w");
if (fout == nullptr)
ERROR_RETURN("fdopen", 1);
should_close = true;
}
BAN::ScopeGuard _([fout, should_close] { if (should_close) fclose(fout); });
if (args.front() == "clear"sv)
{
fprintf(stdout, "\e[H\e[J");
fflush(stdout);
return 0;
fprintf(fout, "\e[H\e[J");
fflush(fout);
}
else if (args.front() == "exit"sv)
{
@ -265,7 +291,7 @@ int execute_command(BAN::Vector<BAN::String>& args)
{
char** current = environ;
while (*current)
printf("%s\n", *current++);
fprintf(fout, "%s\n", *current++);
}
else if (args.front() == "page-fault-test"sv)
{
@ -277,7 +303,7 @@ int execute_command(BAN::Vector<BAN::String>& args)
pid_t pid = fork();
if (pid == 0)
{
printf("child\n");
fprintf(fout, "child\n");
for (;;);
}
if (pid == -1)
@ -298,6 +324,7 @@ int execute_command(BAN::Vector<BAN::String>& args)
pid_t pid = fork();
if (pid == 0)
{
dup2(fileno(fout), STDOUT_FILENO);
if (signal(SIGSEGV, [](int) { printf("SIGSEGV\n"); }) == SIG_ERR)
{
perror("signal");
@ -328,18 +355,18 @@ int execute_command(BAN::Vector<BAN::String>& args)
}
else if (args.front() == "printf-test"sv)
{
printf(" 0.0: %f\n", 0.0f);
printf(" 123.0: %f\n", 123.0f);
printf(" 0.123: %f\n", 0.123f);
printf(" NAN: %f\n", NAN);
printf("+INF: %f\n", INFINITY);
printf("-INF: %f\n", -INFINITY);
fprintf(fout, " 0.0: %f\n", 0.0f);
fprintf(fout, " 123.0: %f\n", 123.0f);
fprintf(fout, " 0.123: %f\n", 0.123f);
fprintf(fout, " NAN: %f\n", NAN);
fprintf(fout, "+INF: %f\n", INFINITY);
fprintf(fout, "-INF: %f\n", -INFINITY);
}
else if (args.front() == "cd"sv)
{
if (args.size() > 2)
{
printf("cd: too many arguments\n");
fprintf(fout, "cd: too many arguments\n");
return 1;
}
@ -367,7 +394,7 @@ int execute_command(BAN::Vector<BAN::String>& args)
if (clock_gettime(CLOCK_MONOTONIC, &start) == -1)
ERROR_RETURN("clock_gettime", 1);
int ret = execute_command(args);
int ret = execute_command(args, fd_in, fd_out);
if (clock_gettime(CLOCK_MONOTONIC, &end) == -1)
ERROR_RETURN("clock_gettime", 1);
@ -379,45 +406,154 @@ int execute_command(BAN::Vector<BAN::String>& args)
int secs = total_ns / 1'000'000'000;
int msecs = (total_ns % 1'000'000'000) / 1'000'000;
printf("took %d.%03d s\n", secs, msecs);
fprintf(fout, "took %d.%03d s\n", secs, msecs);
return ret;
}
else
{
BAN::Vector<char*> cmd_args;
MUST(cmd_args.reserve(args.size() + 1));
for (const auto& arg : args)
MUST(cmd_args.push_back((char*)arg.data()));
MUST(cmd_args.push_back(nullptr));
return {};
}
pid_t pid = fork();
if (pid == 0)
return 0;
}
pid_t execute_command_no_wait(BAN::Vector<BAN::String>& args, int fd_in, int fd_out)
{
if (args.empty())
return 0;
BAN::Vector<char*> cmd_args;
MUST(cmd_args.reserve(args.size() + 1));
for (const auto& arg : args)
MUST(cmd_args.push_back((char*)arg.data()));
MUST(cmd_args.push_back(nullptr));
pid_t pid = fork();
if (pid == 0)
{
if (fd_in != STDIN_FILENO)
{
execv(cmd_args.front(), cmd_args.data());
perror("execv");
exit(1);
if (dup2(fd_in, STDIN_FILENO) == -1)
{
perror("dup2");
exit(1);
}
close(fd_in);
}
if (fd_out != STDOUT_FILENO)
{
if (dup2(fd_out, STDOUT_FILENO) == -1)
{
perror("dup2");
exit(1);
}
close(fd_out);
}
if (pid == -1)
ERROR_RETURN("fork", 1);
if (tcsetpgrp(0, pid) == -1)
ERROR_RETURN("tcsetpgrp", 1);
execv(cmd_args.front(), cmd_args.data());
perror("execv");
exit(1);
}
return pid;
}
int execute_command(BAN::Vector<BAN::String>& args, int fd_in, int fd_out)
{
pid_t pid = execute_command_no_wait(args, fd_in, fd_out);
if (pid == -1)
ERROR_RETURN("fork", 1);
if (tcsetpgrp(0, pid) == -1)
ERROR_RETURN("tcsetpgrp", 1);
int status;
if (waitpid(pid, &status, 0) == -1)
ERROR_RETURN("waitpid", 1);
if (tcsetpgrp(0, getpid()) == -1)
ERROR_RETURN("tcsetpgrp", 1);
if (WIFSIGNALED(status))
fprintf(stderr, "Terminated by signal %d\n", WTERMSIG(status));
return WEXITSTATUS(status);
}
int execute_piped_commands(BAN::Vector<BAN::Vector<BAN::String>>& commands)
{
if (commands.empty())
return 0;
if (commands.size() == 1)
{
auto& command = commands.front();
if (auto ret = execute_builtin(command, STDIN_FILENO, STDOUT_FILENO); ret.has_value())
return ret.value();
return execute_command(command, STDIN_FILENO, STDOUT_FILENO);
}
BAN::Vector<int> exit_codes(commands.size(), 0);
BAN::Vector<pid_t> processes(commands.size(), -1);
int next_stdin = STDIN_FILENO;
for (size_t i = 0; i < commands.size(); i++)
{
bool first = (i == 0);
bool last = (i == commands.size() - 1);
int pipefd[2] { -1, STDOUT_FILENO };
if (!last && pipe(pipefd) == -1)
{
if (i > 0)
close(next_stdin);
perror("pipe");
break;
}
auto builtin_ret = execute_builtin(commands[i], next_stdin, pipefd[1]);
if (builtin_ret.has_value())
exit_codes[i] = builtin_ret.value();
else
{
pid_t pid = execute_command_no_wait(commands[i], next_stdin, pipefd[1]);
processes[i] = pid;
if (first && tcsetpgrp(0, pid) == -1)
ERROR_RETURN("tcsetpgrp", 1);
}
if (next_stdin != STDIN_FILENO)
close(next_stdin);
if (pipefd[1] != STDOUT_FILENO)
close(pipefd[1]);
next_stdin = pipefd[0];
}
for (size_t i = 0; i < commands.size(); i++)
{
if (processes[i] == -1)
continue;
int status;
if (waitpid(pid, &status, 0) == -1)
ERROR_RETURN("waitpid", 1);
if (tcsetpgrp(0, getpid()) == -1)
ERROR_RETURN("tcsetpgrp", 1);
if (waitpid(processes[i], &status, 0) == -1)
{
perror("waitpid");
exit_codes[i] = 69420;
continue;
}
if (WIFSIGNALED(status))
fprintf(stderr, "Terminated by signal %d\n", WTERMSIG(status));
return WEXITSTATUS(status);
if (WEXITSTATUS(status))
exit_codes[i] = WEXITSTATUS(status);
}
return 0;
if (tcsetpgrp(0, getpid()) == -1)
ERROR_RETURN("tcsetpgrp", 1);
return exit_codes.back();
}
int character_length(BAN::StringView prompt)
@ -544,8 +680,8 @@ int main(int argc, char** argv)
BAN::String command;
MUST(command.append(argv[2]));
auto arguments = parse_command(command);
return execute_command(arguments);
auto commands = parse_command(command);
return execute_piped_commands(commands);
}
printf("unknown argument '%s'\n", argv[1]);
@ -657,8 +793,8 @@ int main(int argc, char** argv)
if (!buffers[index].empty())
{
tcsetattr(0, TCSANOW, &old_termios);
auto parsed_arguments = parse_command(buffers[index]);
last_return = execute_command(parsed_arguments);
auto commands = parse_command(buffers[index]);
last_return = execute_piped_commands(commands);
tcsetattr(0, TCSANOW, &new_termios);
MUST(history.push_back(buffers[index]));
buffers = history;