Skip to content

Commit

Permalink
Add jl_getch(), use it from getpass()
Browse files Browse the repository at this point in the history
This works around the lack of `getch()` on non-windows platforms, such
that we can use the windows-specific `getpass()` on all platforms.  This
was necessary to prevent breakage from `musl` due to a bad interaction
between `with_fake_pty()` and `getpass()`.
  • Loading branch information
staticfloat committed Jun 10, 2022
1 parent eb52446 commit b05dfd2
Show file tree
Hide file tree
Showing 4 changed files with 39 additions and 10 deletions.
13 changes: 3 additions & 10 deletions base/util.jl
Original file line number Diff line number Diff line change
Expand Up @@ -257,19 +257,19 @@ graphical interface.
"""
function getpass end

if Sys.iswindows()
_getch() = UInt8(ccall(:jl_getch, Cint, ()))
function getpass(input::TTY, output::IO, prompt::AbstractString)
input === stdin || throw(ArgumentError("getpass only works for stdin"))
print(output, prompt, ": ")
flush(output)
s = SecretBuffer()
plen = 0
while true
c = UInt8(ccall(:_getch, Cint, ()))
c = _getch()
if c == 0xff || c == UInt8('\n') || c == UInt8('\r')
break # EOF or return
elseif c == 0x00 || c == 0xe0
ccall(:_getch, Cint, ()) # ignore function/arrow keys
_getch() # ignore function/arrow keys
elseif c == UInt8('\b') && plen > 0
plen -= 1 # delete last character on backspace
elseif !iscntrl(Char(c)) && plen < 128
Expand All @@ -278,13 +278,6 @@ function getpass(input::TTY, output::IO, prompt::AbstractString)
end
return seekstart(s)
end
else
function getpass(input::TTY, output::IO, prompt::AbstractString)
(input === stdin && output === stdout) || throw(ArgumentError("getpass only works for stdin"))
msg = string(prompt, ": ")
unsafe_SecretBuffer!(ccall(:getpass, Cstring, (Cstring,), msg))
end
end

# allow new getpass methods to be defined if stdin has been
# redirected to some custom stream, e.g. in IJulia.
Expand Down
1 change: 1 addition & 0 deletions src/jl_exported_funcs.inc
Original file line number Diff line number Diff line change
Expand Up @@ -199,6 +199,7 @@
XX(jl_generic_function_def) \
XX(jl_gensym) \
XX(jl_getallocationgranularity) \
XX(jl_getch) \
XX(jl_getnameinfo) \
XX(jl_getpagesize) \
XX(jl_get_ARCH) \
Expand Down
1 change: 1 addition & 0 deletions src/julia.h
Original file line number Diff line number Diff line change
Expand Up @@ -2057,6 +2057,7 @@ extern JL_DLLEXPORT JL_STREAM *JL_STDERR;
JL_DLLEXPORT JL_STREAM *jl_stdout_stream(void);
JL_DLLEXPORT JL_STREAM *jl_stdin_stream(void);
JL_DLLEXPORT JL_STREAM *jl_stderr_stream(void);
JL_DLLEXPORT int jl_getch(void);

// showing and std streams
JL_DLLEXPORT void jl_flush_cstdio(void) JL_NOTSAFEPOINT;
Expand Down
34 changes: 34 additions & 0 deletions src/sys.c
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,11 @@
#include <signal.h>
#include <fcntl.h>

// For `struct termios`
#if !defined(_OS_WINDOWS_)
#include <termios.h>
#endif

#include "julia.h"
#include "julia_internal.h"

Expand Down Expand Up @@ -514,6 +519,35 @@ JL_DLLEXPORT JL_STREAM *jl_stdin_stream(void) { return JL_STDIN; }
JL_DLLEXPORT JL_STREAM *jl_stdout_stream(void) { return JL_STDOUT; }
JL_DLLEXPORT JL_STREAM *jl_stderr_stream(void) { return JL_STDERR; }

// terminal workarounds
JL_DLLEXPORT int jl_getch(void) JL_NOTSAFEPOINT
{
#if defined(_OS_WINDOWS_)
// Windows has an actual `_getch()`, use that:
return _getch();
#else
// On all other platforms, we do the POSIX terminal manipulation dance
char buf = 0;
struct termios old = {0};
fflush(stdout);
if(tcgetattr(0, &old) < 0)
jl_error("tcgetattr()");
old.c_lflag &= ~ICANON;
old.c_lflag &= ~ECHO;
old.c_cc[VMIN] = 1;
old.c_cc[VTIME] = 0;
if(tcsetattr(0, TCSANOW, &old) < 0)
jl_error("tcsetattr(TCSANOW)");
if(read(0, &buf, 1) < 0)
jl_error("read()");
old.c_lflag |= ICANON;
old.c_lflag |= ECHO;
if(tcsetattr(0, TCSADRAIN, &old) < 0)
jl_error("tcsetattr(TCSADRAIN)");
return buf;
#endif
}

// -- processor native alignment information --

JL_DLLEXPORT void jl_native_alignment(uint_t *int8align, uint_t *int16align, uint_t *int32align,
Expand Down

0 comments on commit b05dfd2

Please sign in to comment.