aboutsummaryrefslogtreecommitdiff
path: root/alacritty_terminal/src/tty/unix.rs
diff options
context:
space:
mode:
Diffstat (limited to 'alacritty_terminal/src/tty/unix.rs')
-rw-r--r--alacritty_terminal/src/tty/unix.rs67
1 files changed, 39 insertions, 28 deletions
diff --git a/alacritty_terminal/src/tty/unix.rs b/alacritty_terminal/src/tty/unix.rs
index 078621df..f52f0920 100644
--- a/alacritty_terminal/src/tty/unix.rs
+++ b/alacritty_terminal/src/tty/unix.rs
@@ -1,8 +1,5 @@
//! TTY related functionality.
-use std::borrow::Cow;
-#[cfg(not(target_os = "macos"))]
-use std::env;
use std::ffi::CStr;
use std::fs::File;
use std::io::{Error, ErrorKind, Result};
@@ -10,7 +7,7 @@ use std::mem::MaybeUninit;
use std::os::unix::io::{AsRawFd, FromRawFd, RawFd};
use std::os::unix::process::CommandExt;
use std::process::{Child, Command, Stdio};
-use std::ptr;
+use std::{env, ptr};
use libc::{self, c_int, winsize, TIOCSCTTY};
use log::error;
@@ -21,7 +18,7 @@ use nix::sys::termios::{self, InputFlags, SetArg};
use signal_hook::consts as sigconsts;
use signal_hook_mio::v0_6::Signals;
-use crate::config::{Program, PtyConfig};
+use crate::config::PtyConfig;
use crate::event::{OnResize, WindowSize};
use crate::tty::{ChildEvent, EventedPty, EventedReadWrite};
@@ -121,17 +118,33 @@ impl Pty {
}
}
-#[cfg(target_os = "macos")]
-fn default_shell(pw: &Passwd<'_>) -> Program {
- let shell_name = pw.shell.rsplit('/').next().unwrap();
- let argv = vec![String::from("-c"), format!("exec -a -{} {}", shell_name, pw.shell)];
-
- Program::WithArgs { program: "/bin/bash".to_owned(), args: argv }
+/// Look for a shell in the `$SHELL` environment variable, then in `passwd`.
+fn default_shell(pw: &Passwd<'_>) -> String {
+ env::var("SHELL").unwrap_or_else(|_| pw.shell.to_owned())
}
#[cfg(not(target_os = "macos"))]
-fn default_shell(pw: &Passwd<'_>) -> Program {
- Program::Just(env::var("SHELL").unwrap_or_else(|_| pw.shell.to_owned()))
+fn default_shell_command(pw: &Passwd<'_>) -> Command {
+ Command::new(default_shell(pw))
+}
+
+#[cfg(target_os = "macos")]
+fn default_shell_command(pw: &Passwd<'_>) -> Command {
+ let shell = default_shell(pw);
+ let shell_name = shell.rsplit('/').next().unwrap();
+
+ // On macOS, use the `login` command so the shell will appear as a tty session.
+ let mut login_command = Command::new("/usr/bin/login");
+
+ // Exec the shell with argv[0] prepended by '-' so it becomes a login shell.
+ // `login` normally does this itself, but `-l` disables this.
+ let exec = format!("exec -a -{} {}", shell_name, shell);
+
+ // -f: Bypasses authentication for the already-logged-in user.
+ // -l: Skips changing directory to $HOME and prepending '-' to argv[0].
+ // -p: Preserves the environment.
+ login_command.args(["-flp", pw.name, "/bin/sh", "-c", &exec]);
+ login_command
}
/// Create a new TTY and return a handle to interact with it.
@@ -148,16 +161,14 @@ pub fn new(config: &PtyConfig, window_size: WindowSize, window_id: Option<usize>
let mut buf = [0; 1024];
let pw = get_pw_entry(&mut buf)?;
- let shell = match config.shell.as_ref() {
- Some(shell) => Cow::Borrowed(shell),
- None => Cow::Owned(default_shell(&pw)),
+ let mut builder = if let Some(shell) = config.shell.as_ref() {
+ let mut cmd = Command::new(shell.program());
+ cmd.args(shell.args());
+ cmd
+ } else {
+ default_shell_command(&pw)
};
- let mut builder = Command::new(shell.program());
- for arg in shell.args() {
- builder.arg(arg);
- }
-
// Setup child stdin/stdout/stderr as slave fd of PTY.
// Ownership of fd is transferred to the Stdio structs and will be closed by them at the end of
// this scope. (It is not an issue that the fd is closed three times since File::drop ignores
@@ -171,10 +182,6 @@ pub fn new(config: &PtyConfig, window_size: WindowSize, window_id: Option<usize>
builder.env("USER", pw.name);
builder.env("HOME", pw.dir);
- // Set $SHELL environment variable on macOS, since login does not do it for us.
- #[cfg(target_os = "macos")]
- builder.env("SHELL", config.shell.as_ref().map_or(pw.shell, Program::program));
-
if let Some(window_id) = window_id {
builder.env("WINDOWID", format!("{}", window_id));
}
@@ -210,7 +217,7 @@ pub fn new(config: &PtyConfig, window_size: WindowSize, window_id: Option<usize>
}
// Prepare signal handling before spawning child.
- let signals = Signals::new(&[sigconsts::SIGCHLD]).expect("error preparing signal handling");
+ let signals = Signals::new([sigconsts::SIGCHLD]).expect("error preparing signal handling");
match builder.spawn() {
Ok(child) => {
@@ -231,8 +238,12 @@ pub fn new(config: &PtyConfig, window_size: WindowSize, window_id: Option<usize>
Ok(pty)
},
Err(err) => Err(Error::new(
- ErrorKind::NotFound,
- format!("Failed to spawn command '{}': {}", shell.program(), err),
+ err.kind(),
+ format!(
+ "Failed to spawn command '{}': {}",
+ builder.get_program().to_string_lossy(),
+ err
+ ),
)),
}
}