glibc/sysdeps/m68k/fpu/switch/switch.c

88 lines
3.0 KiB
C
Raw Normal View History

1997-01-20 02:49:42 +00:00
/* Copyright (C) 1991, 1992, 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library.
1991-06-12 18:33:01 +00:00
1997-01-20 02:49:42 +00:00
The GNU C Library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public License as
published by the Free Software Foundation; either version 2 of the
License, or (at your option) any later version.
1991-06-12 18:33:01 +00:00
1997-01-20 02:49:42 +00:00
The GNU C Library 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
Library General Public License for more details.
1991-06-12 18:33:01 +00:00
1997-01-20 02:49:42 +00:00
You should have received a copy of the GNU Library General Public
License along with the GNU C Library; see the file COPYING.LIB. If not,
write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
Boston, MA 02111-1307, USA. */
1991-06-12 18:33:01 +00:00
#include <signal.h>
1992-05-01 18:47:39 +00:00
#include <68881-sw.h>
1991-06-12 18:33:01 +00:00
/* The signal that is sent when a 68881 instruction
is executed and there is no 68881. */
#ifndef TRAPSIG
#define TRAPSIG SIGILL
#endif
/* Zero if no 68881, one if we have a 68881, or -1 if we don't know yet. */
static int have_fpu = -1;
1991-06-12 18:33:01 +00:00
/* Signal handler for the trap that happens if we don't have a 68881. */
static void
1997-01-20 02:49:42 +00:00
trap (sig)
int sig;
1991-06-12 18:33:01 +00:00
{
have_fpu = 0;
}
/* This function is called by functions that want to switch.
The calling function must be a `struct switch_caller' in data space.
It determines whether a 68881 is present, and modifies its caller
to be a static jump to either the 68881 version or the soft version.
It then returns into the function it has chosen to do the work. */
void
1997-01-20 02:49:42 +00:00
__68881_switch (dummy)
int dummy;
1991-06-12 18:33:01 +00:00
{
1997-01-20 02:49:42 +00:00
void **return_address_location = &((void **) &dummy)[-1];
struct switch_caller *const caller
1991-06-12 18:33:01 +00:00
= (struct switch_caller *) (((short int *) *return_address_location) - 1);
if (have_fpu < 0)
1991-06-12 18:33:01 +00:00
{
/* Figure out whether or not we have a 68881. */
__sighandler_t handler = signal (TRAPSIG, trap);
1991-06-12 18:33:01 +00:00
if (handler == SIG_ERR)
/* We can't figure it out, so assume we don't have a 68881.
This assumption will never cause us any problems other than
lost performance, while the reverse assumption could cause
the program to crash. */
have_fpu = 0;
else
{
/* We set `have_fpu' to nonzero, and then execute a 68881
no-op instruction. If we have a 68881, this will do nothing.
If we don't have one, this will trap and the signal handler
will clear `have_fpu'. */
have_fpu = 1;
asm ("fnop");
1991-06-12 18:33:01 +00:00
/* Restore the old signal handler. */
(void) signal (TRAPSIG, handler);
1991-06-12 18:33:01 +00:00
}
}
/* Modify the caller to be a jump to the appropriate address. */
caller->insn = JMP;
caller->target = have_fpu ? caller->fpu : caller->soft;
/* Make the address we will return to be the target we have chosen.
Our return will match the `jsr' done by the caller we have
just modified, and it will be just as if that had instead
been a `jmp' to the new target. */
*return_address_location = caller->target;
}