blob: 6e0ae3150a3491f3a27e8aaa4ca2c79e440d1a87 [file] [log] [blame]
/* do not edit automatically generated by mc from IO. */
/* IO.mod provides Read, Write, Errors procedures mapping onto 0, 1 and 2.
Copyright (C) 2001-2025 Free Software Foundation, Inc.
Contributed by Gaius Mulley <gaius.mulley@southwales.ac.uk>.
This file is part of GNU Modula-2.
GNU Modula-2 is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3, or (at your option)
any later version.
GNU Modula-2 is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
Under Section 7 of GPL version 3, you are granted additional
permissions described in the GCC Runtime Library Exception, version
3.1, as published by the Free Software Foundation.
You should have received a copy of the GNU General Public License and
a copy of the GCC Runtime Library Exception along with this program;
see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
<http://www.gnu.org/licenses/>. */
#include "config.h"
#include "system.h"
#include <stdbool.h>
# if !defined (PROC_D)
# define PROC_D
typedef void (*PROC_t) (void);
typedef struct { PROC_t proc; } PROC;
# endif
# if !defined (TRUE)
# define TRUE (1==1)
# endif
# if !defined (FALSE)
# define FALSE (1==0)
# endif
#define _IO_C
#include "GIO.h"
# include "GStrLib.h"
# include "GSYSTEM.h"
# include "Glibc.h"
# include "GFIO.h"
# include "Gerrno.h"
# include "GASCII.h"
# include "Gtermios.h"
# define MaxDefaultFd 2
typedef struct IO_BasicFds_r IO_BasicFds;
typedef struct IO__T1_a IO__T1;
struct IO_BasicFds_r {
bool IsEof;
bool IsRaw;
};
struct IO__T1_a { IO_BasicFds array[MaxDefaultFd+1]; };
static IO__T1 fdState;
/*
IsDefaultFd - returns TRUE if, fd, is 0, 1 or 2.
*/
extern "C" void IO_Read (char *ch);
/*
doWrite - performs the write of a single character, ch,
onto fd or f.
*/
extern "C" void IO_Write (char ch);
/*
doWrite - performs the write of a single character, ch,
onto fd or f.
*/
extern "C" void IO_Error (char ch);
extern "C" void IO_UnBufferedMode (int fd, bool input);
extern "C" void IO_BufferedMode (int fd, bool input);
/*
EchoOn - turns on echoing for file descriptor, fd. This
only really makes sence for a file descriptor opened
for terminal input or maybe some specific file descriptor
which is attached to a particular piece of hardware.
*/
extern "C" void IO_EchoOn (int fd, bool input);
/*
EchoOff - turns off echoing for file descriptor, fd. This
only really makes sence for a file descriptor opened
for terminal input or maybe some specific file descriptor
which is attached to a particular piece of hardware.
*/
extern "C" void IO_EchoOff (int fd, bool input);
/*
IsDefaultFd - returns TRUE if, fd, is 0, 1 or 2.
*/
static bool IsDefaultFd (int fd);
/*
doWrite - performs the write of a single character, ch,
onto fd or f.
*/
static void doWrite (int fd, FIO_File f, char ch);
/*
setFlag - sets or unsets the appropriate flag in, t.
*/
static void setFlag (termios_TERMIOS t, termios_Flag f, bool b);
/*
doraw - sets all the flags associated with making this
file descriptor into raw input/output.
*/
static void doraw (termios_TERMIOS term);
/*
dononraw - sets all the flags associated with making this
file descriptor into non raw input/output.
*/
static void dononraw (termios_TERMIOS term);
/*
Init -
*/
static void Init (void);
/*
IsDefaultFd - returns TRUE if, fd, is 0, 1 or 2.
*/
static bool IsDefaultFd (int fd)
{
return (fd <= MaxDefaultFd) && (fd >= 0);
/* static analysis guarentees a RETURN statement will be used before here. */
__builtin_unreachable ();
}
/*
doWrite - performs the write of a single character, ch,
onto fd or f.
*/
static void doWrite (int fd, FIO_File f, char ch)
{
int r;
if (fdState.array[fd].IsRaw)
{
/* avoid dangling else. */
if (! fdState.array[fd].IsEof)
{
for (;;)
{
r = static_cast<int> (libc_write (FIO_GetUnixFileDescriptor (f), &ch, static_cast<size_t> (1)));
if (r == 1)
{
return;
}
else if (r == -1)
{
/* avoid dangling else. */
r = errno_geterrno ();
if ((r != errno_EAGAIN) && (r != errno_EINTR))
{
fdState.array[fd].IsEof = true;
return;
}
}
}
}
}
else
{
FIO_WriteChar (f, ch);
}
}
/*
setFlag - sets or unsets the appropriate flag in, t.
*/
static void setFlag (termios_TERMIOS t, termios_Flag f, bool b)
{
if (termios_SetFlag (t, f, b))
{} /* empty. */
}
/*
doraw - sets all the flags associated with making this
file descriptor into raw input/output.
*/
static void doraw (termios_TERMIOS term)
{
/*
* from man 3 termios
* termios_p->c_iflag &= ~(IGNBRK | BRKINT | PARMRK | ISTRIP
* | INLCR | IGNCR | ICRNL | IXON);
* termios_p->c_oflag &= ~OPOST;
* termios_p->c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
* termios_p->c_cflag &= ~(CSIZE | PARENB);
* termios_p->c_cflag |= CS8;
*/
setFlag (term, termios_ignbrk, false);
setFlag (term, termios_ibrkint, false);
setFlag (term, termios_iparmrk, false);
setFlag (term, termios_istrip, false);
setFlag (term, termios_inlcr, false);
setFlag (term, termios_igncr, false);
setFlag (term, termios_icrnl, false);
setFlag (term, termios_ixon, false);
setFlag (term, termios_opost, false);
setFlag (term, termios_lecho, false);
setFlag (term, termios_lechonl, false);
setFlag (term, termios_licanon, false);
setFlag (term, termios_lisig, false);
setFlag (term, termios_liexten, false);
setFlag (term, termios_parenb, false);
setFlag (term, termios_cs8, true);
}
/*
dononraw - sets all the flags associated with making this
file descriptor into non raw input/output.
*/
static void dononraw (termios_TERMIOS term)
{
/*
* we undo these settings, (although we leave the character size alone)
*
* from man 3 termios
* termios_p->c_iflag &= ~(IGNBRK | BRKINT | PARMRK | ISTRIP
* | INLCR | IGNCR | ICRNL | IXON);
* termios_p->c_oflag &= ~OPOST;
* termios_p->c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
* termios_p->c_cflag &= ~(CSIZE | PARENB);
* termios_p->c_cflag |= CS8;
*/
setFlag (term, termios_ignbrk, true);
setFlag (term, termios_ibrkint, true);
setFlag (term, termios_iparmrk, true);
setFlag (term, termios_istrip, true);
setFlag (term, termios_inlcr, true);
setFlag (term, termios_igncr, true);
setFlag (term, termios_icrnl, true);
setFlag (term, termios_ixon, true);
setFlag (term, termios_opost, true);
setFlag (term, termios_lecho, true);
setFlag (term, termios_lechonl, true);
setFlag (term, termios_licanon, true);
setFlag (term, termios_lisig, true);
setFlag (term, termios_liexten, true);
}
/*
Init -
*/
static void Init (void)
{
unsigned int fdi;
for (fdi=0; fdi<=MaxDefaultFd; fdi++)
{
fdState.array[fdi].IsEof = false;
fdState.array[fdi].IsRaw = false;
}
}
/*
IsDefaultFd - returns TRUE if, fd, is 0, 1 or 2.
*/
extern "C" void IO_Read (char *ch)
{
int r;
FIO_FlushBuffer (FIO_StdOut);
FIO_FlushBuffer (FIO_StdErr);
if (fdState.array[0].IsRaw)
{
if (fdState.array[0].IsEof)
{
(*ch) = ASCII_eof;
}
else
{
for (;;)
{
r = static_cast<int> (libc_read (FIO_GetUnixFileDescriptor (FIO_StdIn), ch, static_cast<size_t> (1)));
if (r == 1)
{
return;
}
else if (r == -1)
{
/* avoid dangling else. */
r = errno_geterrno ();
if (r != errno_EAGAIN)
{
fdState.array[0].IsEof = true;
(*ch) = ASCII_eof;
return;
}
}
}
}
}
else
{
(*ch) = FIO_ReadChar (FIO_StdIn);
}
}
/*
doWrite - performs the write of a single character, ch,
onto fd or f.
*/
extern "C" void IO_Write (char ch)
{
doWrite (1, FIO_StdOut, ch);
}
/*
doWrite - performs the write of a single character, ch,
onto fd or f.
*/
extern "C" void IO_Error (char ch)
{
doWrite (2, FIO_StdErr, ch);
}
extern "C" void IO_UnBufferedMode (int fd, bool input)
{
termios_TERMIOS term;
int result;
if (IsDefaultFd (fd))
{
fdState.array[fd].IsRaw = true;
}
term = termios_InitTermios ();
if ((termios_tcgetattr (fd, term)) == 0)
{
doraw (term);
if (input)
{
result = termios_tcsetattr (fd, termios_tcsflush (), term);
}
else
{
result = termios_tcsetattr (fd, termios_tcsdrain (), term);
}
}
term = termios_KillTermios (term);
}
extern "C" void IO_BufferedMode (int fd, bool input)
{
termios_TERMIOS term;
int r;
if (IsDefaultFd (fd))
{
fdState.array[fd].IsRaw = false;
}
term = termios_InitTermios ();
if ((termios_tcgetattr (fd, term)) == 0)
{
dononraw (term);
if (input)
{
r = termios_tcsetattr (fd, termios_tcsflush (), term);
}
else
{
r = termios_tcsetattr (fd, termios_tcsdrain (), term);
}
}
term = termios_KillTermios (term);
}
/*
EchoOn - turns on echoing for file descriptor, fd. This
only really makes sence for a file descriptor opened
for terminal input or maybe some specific file descriptor
which is attached to a particular piece of hardware.
*/
extern "C" void IO_EchoOn (int fd, bool input)
{
termios_TERMIOS term;
int result;
term = termios_InitTermios ();
if ((termios_tcgetattr (fd, term)) == 0)
{
setFlag (term, termios_lecho, true);
if (input)
{
result = termios_tcsetattr (fd, termios_tcsflush (), term);
}
else
{
result = termios_tcsetattr (fd, termios_tcsdrain (), term);
}
}
term = termios_KillTermios (term);
}
/*
EchoOff - turns off echoing for file descriptor, fd. This
only really makes sence for a file descriptor opened
for terminal input or maybe some specific file descriptor
which is attached to a particular piece of hardware.
*/
extern "C" void IO_EchoOff (int fd, bool input)
{
termios_TERMIOS term;
int result;
term = termios_InitTermios ();
if ((termios_tcgetattr (fd, term)) == 0)
{
setFlag (term, termios_lecho, false);
if (input)
{
result = termios_tcsetattr (fd, termios_tcsflush (), term);
}
else
{
result = termios_tcsetattr (fd, termios_tcsdrain (), term);
}
}
term = termios_KillTermios (term);
}
extern "C" void _M2_IO_init (__attribute__((unused)) int argc, __attribute__((unused)) char *argv[], __attribute__((unused)) char *envp[])
{
Init ();
}
extern "C" void _M2_IO_fini (__attribute__((unused)) int argc, __attribute__((unused)) char *argv[], __attribute__((unused)) char *envp[])
{
}