tmux/osdep-win32.c
Rafael Kitover 37b27d2eef
Support use as a Windows program outside of Cygwin
If `getenv("SHELL")` is `NULL`, use the function `win32_setenv_shell()`
added to `osdep-win32-cpp.cpp` to set it using the command line of the
parent process, which is retrieved using WMI WIN32 API.

This is a C++ file because the OLE/WMI API is only available for C++.
Adjust the autotools code to add this file and link the necessary
Windows DLLs.

Include some fixed/missing MINGW headers necessary to compile this file.
This will be fixed in the relevant places and they will be removed.

Add a new macro WIN32_PLATFORM for Windows specific functionality, currently
Cygwin and MSYS2, in the future for the native port as well.

When spawning commands using the shell, check for cmd.exe on Windows and
use the `/c` switch, otherwise use `-c` which works for PowerShell,
MSYS2, Cygwin and Git Bash etc..

Adjust code that uses `/tmp/` to use
`$env:USERPROFILE/AppData/Local/Temp/` outside of a Cygwin virtual
filesystem when `/tmp/` is not available, add the function
`win32_get_tmpdir()` and related functions to `osdep-win32.c` for this.

Use `getenv("USERPROFILE")` when `getenv("HOME")` is `NULL`.

When outside of a Cygwin virtual filesystem, use
`C:\ProgramData\tmux\tmux.conf:$USERPROFILE\.tmux.conf:$LOCALAPPDATA\tmux\tmux.conf`
as the config search order.

Use the ncurses term-driver with `TERM="#win32con"` when a terminfo
database is not available. This will require patches to ncurses as well
as MSYS2 and Cygwin to work.

Signed-off-by: Rafael Kitover <rkitover@gmail.com>
2024-10-02 16:01:39 +00:00

102 lines
1.9 KiB
C

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "tmux.h"
#include "xmalloc.h"
static int in_cygwin_virtual_filesystem(void);
#define PROFILE_TMPDIR "/AppData/Local/Temp/"
static char tmpdir[4096] = {0};
static char socket_dir[4096] = {0};
static char tmpfile_template[4096] = {0};
static int in_cygwin_fs = -1;
static int
in_cygwin_virtual_filesystem(void)
{
if (in_cygwin_fs != -1)
return in_cygwin_fs;
in_cygwin_fs = 0;
if (
(access("/bin/sh", F_OK) == 0) &&
(access("/usr", F_OK) == 0) &&
(access("/home", F_OK) == 0) &&
(access("/proc", F_OK) == 0) &&
(access("/var", F_OK) == 0) &&
(access("/tmp", F_OK) == 0) &&
(access("/dev/null", F_OK) == 0) &&
(access("/dev/random", F_OK) == 0) &&
(access("/dev/stdout", F_OK) == 0) &&
(access("/proc/stat", F_OK) == 0)
) {
in_cygwin_fs = 1;
}
return in_cygwin_fs;
}
const char *
win32_get_tmpdir(void)
{
char *profile_dir = NULL;
if (*tmpdir)
return tmpdir;
if (in_cygwin_virtual_filesystem()) {
strncpy(tmpdir, "/tmp/", 4096);
return tmpdir;
}
if ((profile_dir = getenv("USERPROFILE")))
snprintf(tmpdir, 4096, "%s%s", profile_dir, PROFILE_TMPDIR);
return tmpdir;
}
const char *
win32_get_socket_dir_search_path(void)
{
if (*socket_dir)
return socket_dir;
snprintf(socket_dir, 4096, "$TMUX_TMPDIR:%s", win32_get_tmpdir());
return socket_dir;
}
const char *
win32_get_conf_search_path(void)
{
if (in_cygwin_virtual_filesystem())
return TMUX_CONF;
return TMUX_CONF_WIN32;
}
const char *
win32_get_tmpfile_template(void)
{
if (*tmpfile_template)
return tmpfile_template;
snprintf(tmpfile_template, 4096, "%s/tmux.XXXXXXXX", win32_get_tmpdir());
return tmpfile_template;
}
const char *
win32_get_shell_cmd_switch(const char *shell)
{
if (strstr(shell, "cmd.exe") != NULL)
return "/c";
else
return "-c";
}