gcc/libgm2/libm2pim/Selective.cc
Iain Sandoe ec6d1fc09f modula-2: Fix registration of modules via constructors [PR108183].
This reworks the mechanism used for module registration to use init-
time constructors.  The order of registration is not important, the
actual initialization dependency tree will be computed early in the
execution (all that matters is that we have registered before that).

This fixes a potential issue in which the external name known to the
m2 system is of the form _M2_XXXXXX_ctor() but the C++ code was
producing a static variable instance with the same name.

Signed-off-by: Iain Sandoe <iain@sandoe.co.uk>

	PR modula2/108183

gcc/m2/ChangeLog:

	* gm2-libs-ch/UnixArgs.cc (_M2_UnixArgs_ctor): Rework to use
	an extern "C" function with 'constructor' attribute.
	* gm2-libs-ch/dtoa.cc (_M2_dtoa_ctor): Likewise.
	* gm2-libs-ch/ldtoa.cc (_M2_ldtoa_ctor): Likewise.

libgm2/ChangeLog:

	* libm2cor/KeyBoardLEDs.cc (_M2_KeyBoardLEDs_ctor): Rework to use
	an extern "C" function with 'constructor' attribute.
	* libm2iso/ErrnoCategory.cc (_M2_ErrnoCategory_ctor): Likewise.
	* libm2iso/RTco.cc (_M2_RTco_ctor): Likewise.
	* libm2pim/Selective.cc (_M2_Selective_ctor): Likewise.
	* libm2pim/SysExceptions.cc (_M2_SysExceptions_ctor): Likewise.
	* libm2pim/UnixArgs.cc (_M2_UnixArgs_ctor): Likewise.
	* libm2pim/cgetopt.cc (_M2_cgetopt_ctor): Likewise.
	* libm2pim/dtoa.cc (_M2_dtoa_ctor): Likewise.
	* libm2pim/errno.cc (_M2_errno_ctor): Likewise.
	* libm2pim/ldtoa.cc (_M2_ldtoa_ctor): Likewise.
	* libm2pim/sckt.cc (_M2_sckt_ctor): Likewise.
	* libm2pim/termios.cc (_M2_termios_ctor): Likewise.
	* libm2pim/wrapc.c: Add a new line to the file end.
2023-01-04 14:54:41 +00:00

318 lines
6 KiB
C++

/* Selective.c provide access to timeval and select.
Copyright (C) 2009-2022 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 <m2rts.h>
#if defined(HAVE_STDDEF_H)
/* Obtain a definition for NULL. */
#include <stddef.h>
#endif
#if defined(HAVE_STDIO_H)
/* Obtain a definition for NULL. */
#include <stdio.h>
#endif
#if defined(HAVE_SYS_TIME_H)
#include <sys/time.h>
#endif
#if defined(HAVE_TIME_H)
/* Obtain a definition for NULL. */
#include <time.h>
#endif
#if defined(HAVE_STRING_H)
/* Obtain a definition for NULL. */
#include <string.h>
#endif
#if defined(HAVE_WCHAR_H)
/* Obtain a definition for NULL. */
#include <wchar.h>
#endif
#if defined(HAVE_STDLIB_H)
/* Obtain a prototype for free and malloc. */
#include <stdlib.h>
#endif
#if defined(HAVE_SYS_TYPES_H)
#include <sys/types.h>
#endif
#if defined(HAVE_UNISTD_H)
#include <unistd.h>
#endif
#if !defined(NULL)
#define NULL (void *)0
#endif
#if defined(HAVE_SELECT)
#define FDSET_T fd_set
#else
#define FDSET_T void
#endif
/* Select wrap a call to the C select. */
#if defined(HAVE_STRUCT_TIMEVAL)
extern "C" int
Selective_Select (int nooffds, fd_set *readfds, fd_set *writefds,
fd_set *exceptfds, struct timeval *timeout)
{
return select (nooffds, readfds, writefds, exceptfds, timeout);
}
#else
extern "C" int
Selective_Select (int nooffds, void *readfds, void *writefds, void *exceptfds,
void *timeout)
{
return 0;
}
#endif
/* InitTime initializes a timeval structure and returns a pointer to it. */
#if defined(HAVE_STRUCT_TIMEVAL)
extern "C" struct timeval *
Selective_InitTime (unsigned int sec, unsigned int usec)
{
struct timeval *t = (struct timeval *)malloc (sizeof (struct timeval));
t->tv_sec = (long int)sec;
t->tv_usec = (long int)usec;
return t;
}
extern "C" void
Selective_GetTime (struct timeval *t, unsigned int *sec, unsigned int *usec)
{
*sec = (unsigned int)t->tv_sec;
*usec = (unsigned int)t->tv_usec;
}
extern "C" void
Selective_SetTime (struct timeval *t, unsigned int sec, unsigned int usec)
{
t->tv_sec = sec;
t->tv_usec = usec;
}
/* KillTime frees the timeval structure and returns NULL. */
extern "C" struct timeval *
Selective_KillTime (struct timeval *t)
{
#if defined(HAVE_STDLIB_H)
free (t);
#endif
return NULL;
}
/* InitSet returns a pointer to a FD_SET. */
extern "C" FDSET_T *
Selective_InitSet (void)
{
#if defined(HAVE_STDLIB_H)
FDSET_T *s = (FDSET_T *)malloc (sizeof (FDSET_T));
return s;
#else
return NULL
#endif
}
/* KillSet frees the FD_SET and returns NULL. */
extern "C" FDSET_T *
Selective_KillSet (FDSET_T *s)
{
#if defined(HAVE_STDLIB_H)
free (s);
#endif
return NULL;
}
/* FdZero generate an empty set. */
extern "C" void
Selective_FdZero (FDSET_T *s)
{
FD_ZERO (s);
}
/* FS_Set include an element, fd, into set, s. */
extern "C" void
Selective_FdSet (int fd, FDSET_T *s)
{
FD_SET (fd, s);
}
/* FdClr exclude an element, fd, from the set, s. */
extern "C" void
Selective_FdClr (int fd, FDSET_T *s)
{
FD_CLR (fd, s);
}
/* FdIsSet return TRUE if, fd, is present in set, s. */
extern "C" int
Selective_FdIsSet (int fd, FDSET_T *s)
{
return FD_ISSET (fd, s);
}
/* GetTimeOfDay fills in a record, Timeval, filled in with the
current system time in seconds and microseconds.
It returns zero (see man 3p gettimeofday). */
extern "C" int
Selective_GetTimeOfDay (struct timeval *t)
{
return gettimeofday (t, NULL);
}
#else
extern "C" void *
Selective_InitTime (unsigned int sec, unsigned int usec)
{
return NULL;
}
extern "C" void *
Selective_KillTime (void *t)
{
return NULL;
}
extern "C" void
Selective_GetTime (void *t, unsigned int *sec, unsigned int *usec)
{
}
extern "C" void
Selective_SetTime (void *t, unsigned int sec, unsigned int usec)
{
}
extern "C" FDSET_T *
Selective_InitSet (void)
{
return NULL;
}
extern "C" FDSET_T *
Selective_KillSet (void)
{
return NULL;
}
extern "C" void
Selective_FdZero (void *s)
{
}
extern "C" void
Selective_FdSet (int fd, void *s)
{
}
extern "C" void
Selective_FdClr (int fd, void *s)
{
}
extern "C" int
Selective_FdIsSet (int fd, void *s)
{
return 0;
}
extern "C" int
Selective_GetTimeOfDay (void *t)
{
return -1;
}
#endif
/* MaxFdsPlusOne returns max (a + 1, b + 1). */
extern "C" int
Selective_MaxFdsPlusOne (int a, int b)
{
if (a > b)
return a + 1;
else
return b + 1;
}
/* WriteCharRaw writes a single character to the file descriptor. */
extern "C" void
Selective_WriteCharRaw (int fd, char ch)
{
write (fd, &ch, 1);
}
/* ReadCharRaw read and return a single char from file descriptor, fd. */
extern "C" char
Selective_ReadCharRaw (int fd)
{
char ch;
read (fd, &ch, 1);
return ch;
}
extern "C" void
_M2_Selective_init (int argc, char *argv[], char *envp[])
{
}
extern "C" void
_M2_Selective_fini (int argc, char *argv[], char *envp[])
{
}
extern "C" void
_M2_Selective_dep (void)
{
}
extern "C" void __attribute__((__constructor__))
_M2_Selective_ctor (void)
{
M2RTS_RegisterModule ("Selective", _M2_Selective_init, _M2_Selective_fini,
_M2_Selective_dep);
}