mirror of
https://codeberg.org/dnkl/foot.git
synced 2026-04-15 08:21:03 -04:00
slave: exec shell from conf
This commit is contained in:
parent
19aaa7b774
commit
153894eb73
3 changed files with 89 additions and 4 deletions
2
main.c
2
main.c
|
|
@ -548,7 +548,7 @@ main(int argc, char *const *argv)
|
||||||
|
|
||||||
case 0:
|
case 0:
|
||||||
/* Child */
|
/* Child */
|
||||||
slave_spawn(term.ptmx);
|
slave_spawn(term.ptmx, conf.shell);
|
||||||
assert(false);
|
assert(false);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
|
|
||||||
89
slave.c
89
slave.c
|
|
@ -12,12 +12,95 @@
|
||||||
#define LOG_ENABLE_DBG 0
|
#define LOG_ENABLE_DBG 0
|
||||||
#include "log.h"
|
#include "log.h"
|
||||||
|
|
||||||
|
static bool
|
||||||
|
push_argv(char ***argv, size_t *size, char *arg, size_t *argc)
|
||||||
|
{
|
||||||
|
if (arg != NULL && arg[0] == '%')
|
||||||
|
return true;
|
||||||
|
|
||||||
|
if (*argc >= *size) {
|
||||||
|
size_t new_size = *size > 0 ? 2 * *size : 10;
|
||||||
|
char **new_argv = realloc(*argv, new_size * sizeof(new_argv[0]));
|
||||||
|
|
||||||
|
if (new_argv == NULL)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
*argv = new_argv;
|
||||||
|
*size = new_size;
|
||||||
|
}
|
||||||
|
|
||||||
|
(*argv)[(*argc)++] = arg;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool
|
||||||
|
tokenize_cmdline(char *cmdline, char ***argv)
|
||||||
|
{
|
||||||
|
*argv = NULL;
|
||||||
|
size_t argv_size = 0;
|
||||||
|
|
||||||
|
bool first_token_is_quoted = cmdline[0] == '"' || cmdline[0] == '\'';
|
||||||
|
char delim = first_token_is_quoted ? cmdline[0] : ' ';
|
||||||
|
|
||||||
|
char *p = first_token_is_quoted ? &cmdline[1] : &cmdline[0];
|
||||||
|
|
||||||
|
size_t idx = 0;
|
||||||
|
while (*p != '\0') {
|
||||||
|
char *end = strchr(p, delim);
|
||||||
|
if (end == NULL) {
|
||||||
|
if (delim != ' ') {
|
||||||
|
LOG_ERR("unterminated %s quote\n", delim == '"' ? "double" : "single");
|
||||||
|
free(*argv);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!push_argv(argv, &argv_size, p, &idx) ||
|
||||||
|
!push_argv(argv, &argv_size, NULL, &idx))
|
||||||
|
{
|
||||||
|
goto err;
|
||||||
|
} else
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
*end = '\0';
|
||||||
|
|
||||||
|
if (!push_argv(argv, &argv_size, p, &idx))
|
||||||
|
goto err;
|
||||||
|
|
||||||
|
p = end + 1;
|
||||||
|
while (*p == delim)
|
||||||
|
p++;
|
||||||
|
|
||||||
|
while (*p == ' ')
|
||||||
|
p++;
|
||||||
|
|
||||||
|
if (*p == '"' || *p == '\'') {
|
||||||
|
delim = *p;
|
||||||
|
p++;
|
||||||
|
} else
|
||||||
|
delim = ' ';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!push_argv(argv, &argv_size, NULL, &idx))
|
||||||
|
goto err;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
|
||||||
|
err:
|
||||||
|
free(*argv);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
slave_spawn(int ptmx)
|
slave_spawn(int ptmx, char *cmd)
|
||||||
{
|
{
|
||||||
int pts = -1;
|
int pts = -1;
|
||||||
const char *pts_name = ptsname(ptmx);
|
const char *pts_name = ptsname(ptmx);
|
||||||
|
|
||||||
|
char **argv = NULL;
|
||||||
|
if (!tokenize_cmdline(cmd, &argv))
|
||||||
|
goto err;
|
||||||
|
|
||||||
if (grantpt(ptmx) == -1) {
|
if (grantpt(ptmx) == -1) {
|
||||||
LOG_ERRNO("failed to grantpt()");
|
LOG_ERRNO("failed to grantpt()");
|
||||||
goto err;
|
goto err;
|
||||||
|
|
@ -52,9 +135,11 @@ slave_spawn(int ptmx)
|
||||||
close(pts);
|
close(pts);
|
||||||
pts = -1;
|
pts = -1;
|
||||||
|
|
||||||
execl("/usr/bin/zsh", "/usr/bin/zsh", NULL);
|
execvp(argv[0], argv);
|
||||||
|
|
||||||
err:
|
err:
|
||||||
|
if (argv)
|
||||||
|
free(argv);
|
||||||
if (pts != -1)
|
if (pts != -1)
|
||||||
close(pts);
|
close(pts);
|
||||||
if (ptmx != -1)
|
if (ptmx != -1)
|
||||||
|
|
|
||||||
2
slave.h
2
slave.h
|
|
@ -1,4 +1,4 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
|
|
||||||
void slave_spawn(int ptmx);
|
void slave_spawn(int ptmx, char *cmd);
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue