1997-03-23 02:11  Ulrich Drepper  <drepper@cygnus.com>

	* time/sys/time.h: Make values ITIMER_* also available as macros.

	* elf/dl-support.c (_dl_sysdep_read_whole_file): Don't call
	__fstat but instead __fxstat directly to avoid dependency on
	libc.a when inline failed.
	* sysdeps/generic/dl-sysdep.c (_dl_sysdep_read_whole_file): Likewise.

	* math/Makefile (libm-routines): Add s_remquo.
	* sysdeps/libm-i387/s_remquo.S: New file.
	* sysdeps/libm-i387/s_remquof.S: New file.
	* sysdeps/libm-i387/s_remquol.S: New file.
	* sysdeps/libm-ieee754/s_remquo.c: New file.
	* sysdeps/libm-ieee754/s_remquof.c: New file.
	* sysdeps/libm-ieee754/s_remquol.c: New file.

	* math/libm-test.c (remquo_test): New function.

	* sysdeps/libm-ieee754/s_rintl.c: Handle implicit leading one
	correctly.

1997-03-22 14:06  Ulrich Drepper  <drepper@cygnus.com>

	* math/Makefile (libm-calls): Add s_nan.
	* sysdeps/libm-ieee754/s_nan.c: New file.  Implement `nan' function.
	* sysdeps/libm-ieee754/s_nanf.c: New file.  Implement `nanf' function.
	* sysdeps/libm-ieee754/s_nanl.c: New file.  Implement `nanl' function.

	* math/libm-test.c (basic_tests): Add tests for `nan' function.

	* math/libm-test.c (copysign_test): New function.
	(main): Call copysign_test.

1997-03-22 06:28  Ulrich Drepper  <drepper@cygnus.com>

	* sysdeps/libm-ieee754/s_nextafter.c: Return y if x == y.
	* sysdeps/libm-ieee754/s_nextafterf.c: Likewise.
	* sysdeps/libm-ieee754/s_nextafterl.c: Likewise.
	* sysdeps/libm-i387/s_nextafterl.c: Likewise.

	* math/libm-test.c (fdim_test, fmin_test, fmax_test, nextafter_test):
	New functions.  Test these functions.
	(main): Call above new functions.

1997-03-22 04:53  Ulrich Drepper  <drepper@cygnus.com>

	* Net release 2.0.2.

1997-03-22 04:37  Ulrich Drepper  <drepper@cygnus.com>

	* sysdeps/unix/sysv/linux/termbits.h (OXTABS): Define as alias for
	XTABS.
	* sysdeps/unix/sysv/linux/alpha/termbits.h (OXTABS): Likewise.
	* termios/sys/ttydefaults.h: Partly revert patch by Andreas Schwab
	of Sun Dec 15 16:33:44 1996.
	Proposed by Thomas Bushnell <thomas@gnu.ai.mit.edu>.

1997-03-21 13:41  Roland McGrath  <roland@baalperazim.frob.com>

	* sysdeps/sparc/dl-machine.h (elf_machine_rel): Rewritten as for i386.
	Check here for non-SHN_UNDEF STB_LOCAL symbols don't do any lookup or
	consult their values.
	(elf_machine_lookup_noexec_p, elf_machine_lookup_noplt_p,
	ELF_MACHINE_RELOC_NOPLT): New macros.

	* elf/rtld.c (dl_main): Pass ELF_MACHINE_RELOC_NOPLT to
	_dl_lookup_symbol in place of DL_LOOKUP_NOPLT.
	* sysdeps/i386/dl-machine.h (ELF_MACHINE_RELOC_NOPLT): New macro.

	* sysdeps/i386/dl-machine.h (elf_machine_rel): Rewritten to do the
	symbol lookup before checking reloc type except for R_386_RELATIVE.
	(elf_machine_lookup_noexec_p, elf_machine_lookup_noplt_p): New macros.

	* elf/dl-reloc.c (RESOLVE): Remove STB_LOCAL check; let it be
	per-machine since it supposedly can't happen on i386.

	* elf/dl-lookup.c (do_lookup): Change arg FLAGS to RELOC_TYPE.  Use
	elf_machine_lookup_{noexec,noplt}_p macros on it.  Remove gratuitous
	indirection from REF arg; change callers.
	(_dl_lookup_symbol, _dl_lookup_versioned_symbol): Change arg name.
	(_dl_lookup_symbol_skip, _dl_lookup_versioned_symbol_skip): Remove
	FLAGS arg altogether.
	* elf/dlsym.c: Remove argument of FLAGS parameter.
	* elf/dlvsym.c: Likewise.
	* elf/link.h: Update decls.

	* Makefile (distribute): Add BUGS.
This commit is contained in:
Ulrich Drepper 1997-03-23 01:54:07 +00:00
parent 5ae9d168f6
commit bc9f6000f6
36 changed files with 1282 additions and 228 deletions

43
BUGS
View File

@ -1,21 +1,28 @@
List of known bugs (certainly very incomplete) List of known bugs (certainly very incomplete)
---------------------------------------------- ----------------------------------------------
Time-stamp: <1997-03-20T02:33:37+0100 drepper> Time-stamp: <1997-03-22T04:31:41+0100 drepper>
This following list contains those bugs which I'm aware of. Please This following list contains those bugs which I'm aware of. Please
make sure that bugs you report are not listed here. I you can fix one make sure that bugs you report are not listed here. If you can fix one
of these bugs I'll certainly be glad to receive a patch. of these bugs/limitations I'll certainly be glad to receive a patch.
Another source of information about bugs is the problem data base of the
GNU project. There is an easy to use WWW interface is available at
http://pogo.gnu.ai.mit.edu:8080/cgi-bin/wwwgnats.pl
I would appreciate it very much if you could verify the problem was not
reported before by looking though the database. To make the information
in this data as useful as possible please report bugs always using the
`glibcbug' shell script which gets installed with GNU libc.
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Severity: [ *] to [***] Severity: [ *] to [***]
[***] Profiling currently does not work with programs using NSS.
[***] At least on Linux/Alpha, there seem to be problems with dynamically
loading NSS modules in certain situations.
[PR libc/137]
[ **] For GNU libc on Linux, there is still no solution for the UTMP [ **] For GNU libc on Linux, there is still no solution for the UTMP
problem. problem.
[Among others: PR libc/39] [Among others: PR libc/39]
@ -23,7 +30,7 @@ Severity: [ *] to [***]
[ **] There are problems with signal handling when using LinuxThreads. [ **] There are problems with signal handling when using LinuxThreads.
[ **] The `cbrtl' function is inaccurate. The algorithm used for `double' [ **] The `cbrtl' function is inaccurate. The algorithm used for `double'
and `float' is not usable. and `float' is not usable for `long double'.
[ **] Not really a bug, but it could lead to such: [ **] Not really a bug, but it could lead to such:
The RPC code is ugly ugly ugly. It's more or less verbatim taken The RPC code is ugly ugly ugly. It's more or less verbatim taken
@ -37,3 +44,19 @@ Severity: [ *] to [***]
[ *] The syslog function should print to the console if the LOG_CONS [ *] The syslog function should print to the console if the LOG_CONS
flag was given. flag was given.
[PR libc/72] [PR libc/72]
[ *] On Linux, the <linux/posix_types.h> is not clean enough to satisfy
the C++ namespace rules. Declaring `struct fd_set' also makes
`fd_set' available in the global namespace which conflicts with
the definition of `fd_set' in glibc.
[PR libc/79]
[ *] On Linux, there should be a way to prevent defining the symbol
NGROUPS_MAX in the <linux/limits.h> header file. In glibc it
is defined in <posix1_lim.h> which must not make the other
symbols in <linux/limits.h> available.
[PR libc/140]
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Ulrich Drepper
drepper@cygnus.com

View File

@ -1,5 +1,94 @@
1997-03-23 02:11 Ulrich Drepper <drepper@cygnus.com>
* time/sys/time.h: Make values ITIMER_* also available as macros.
* elf/dl-support.c (_dl_sysdep_read_whole_file): Don't call
__fstat but instead __fxstat directly to avoid dependency on
libc.a when inline failed.
* sysdeps/generic/dl-sysdep.c (_dl_sysdep_read_whole_file): Likewise.
* math/Makefile (libm-routines): Add s_remquo.
* sysdeps/libm-i387/s_remquo.S: New file.
* sysdeps/libm-i387/s_remquof.S: New file.
* sysdeps/libm-i387/s_remquol.S: New file.
* sysdeps/libm-ieee754/s_remquo.c: New file.
* sysdeps/libm-ieee754/s_remquof.c: New file.
* sysdeps/libm-ieee754/s_remquol.c: New file.
* math/libm-test.c (remquo_test): New function.
* sysdeps/libm-ieee754/s_rintl.c: Handle implicit leading one
correctly.
1997-03-22 14:06 Ulrich Drepper <drepper@cygnus.com>
* math/Makefile (libm-calls): Add s_nan.
* sysdeps/libm-ieee754/s_nan.c: New file. Implement `nan' function.
* sysdeps/libm-ieee754/s_nanf.c: New file. Implement `nanf' function.
* sysdeps/libm-ieee754/s_nanl.c: New file. Implement `nanl' function.
* math/libm-test.c (basic_tests): Add tests for `nan' function.
* math/libm-test.c (copysign_test): New function.
(main): Call copysign_test.
1997-03-22 06:28 Ulrich Drepper <drepper@cygnus.com>
* sysdeps/libm-ieee754/s_nextafter.c: Return y if x == y.
* sysdeps/libm-ieee754/s_nextafterf.c: Likewise.
* sysdeps/libm-ieee754/s_nextafterl.c: Likewise.
* sysdeps/libm-i387/s_nextafterl.c: Likewise.
* math/libm-test.c (fdim_test, fmin_test, fmax_test, nextafter_test):
New functions. Test these functions.
(main): Call above new functions.
1997-03-22 04:53 Ulrich Drepper <drepper@cygnus.com>
* Net release 2.0.2.
1997-03-22 04:37 Ulrich Drepper <drepper@cygnus.com>
* sysdeps/unix/sysv/linux/termbits.h (OXTABS): Define as alias for
XTABS.
* sysdeps/unix/sysv/linux/alpha/termbits.h (OXTABS): Likewise.
* termios/sys/ttydefaults.h: Partly revert patch by Andreas Schwab
of Sun Dec 15 16:33:44 1996.
Proposed by Thomas Bushnell <thomas@gnu.ai.mit.edu>.
1997-03-21 13:41 Roland McGrath <roland@baalperazim.frob.com>
* sysdeps/sparc/dl-machine.h (elf_machine_rel): Rewritten as for i386.
Check here for non-SHN_UNDEF STB_LOCAL symbols don't do any lookup or
consult their values.
(elf_machine_lookup_noexec_p, elf_machine_lookup_noplt_p,
ELF_MACHINE_RELOC_NOPLT): New macros.
* elf/rtld.c (dl_main): Pass ELF_MACHINE_RELOC_NOPLT to
_dl_lookup_symbol in place of DL_LOOKUP_NOPLT.
* sysdeps/i386/dl-machine.h (ELF_MACHINE_RELOC_NOPLT): New macro.
* sysdeps/i386/dl-machine.h (elf_machine_rel): Rewritten to do the
symbol lookup before checking reloc type except for R_386_RELATIVE.
(elf_machine_lookup_noexec_p, elf_machine_lookup_noplt_p): New macros.
* elf/dl-reloc.c (RESOLVE): Remove STB_LOCAL check; let it be
per-machine since it supposedly can't happen on i386.
* elf/dl-lookup.c (do_lookup): Change arg FLAGS to RELOC_TYPE. Use
elf_machine_lookup_{noexec,noplt}_p macros on it. Remove gratuitous
indirection from REF arg; change callers.
(_dl_lookup_symbol, _dl_lookup_versioned_symbol): Change arg name.
(_dl_lookup_symbol_skip, _dl_lookup_versioned_symbol_skip): Remove
FLAGS arg altogether.
* elf/dlsym.c: Remove argument of FLAGS parameter.
* elf/dlvsym.c: Likewise.
* elf/link.h: Update decls.
1997-03-21 20:55 Ulrich Drepper <drepper@cygnus.com> 1997-03-21 20:55 Ulrich Drepper <drepper@cygnus.com>
* Makefile (distribute): Add BUGS.
* sysdeps/generic/machine-gmon.h: Update copyright. * sysdeps/generic/machine-gmon.h: Update copyright.
* sysdeps/i386/Makefile [$(subdir)=gmon] (sysdep_routines): Add * sysdeps/i386/Makefile [$(subdir)=gmon] (sysdep_routines): Add

View File

@ -290,7 +290,7 @@ $(objpfx)isomac: isomac.c
# Make the distribution tarfile. # Make the distribution tarfile.
distribute := README INSTALL FAQ NOTES NEWS PROJECTS \ distribute := README INSTALL FAQ NOTES NEWS PROJECTS BUGS \
COPYING.LIB COPYING ChangeLog ChangeLog.[0-9] \ COPYING.LIB COPYING ChangeLog ChangeLog.[0-9] \
Makefile Makeconfig Makerules Rules Make-dist MakeTAGS \ Makefile Makeconfig Makerules Rules Make-dist MakeTAGS \
extra-lib.mk o-iterator.mk isomac.c \ extra-lib.mk o-iterator.mk isomac.c \

View File

@ -22,7 +22,8 @@
#include <assert.h> #include <assert.h>
#include <string.h> #include <string.h>
#include <dl-hash.h> #include "dl-hash.h"
#include <dl-machine.h>
#include "../stdio-common/_itoa.h" #include "../stdio-common/_itoa.h"
#define VERSTAG(tag) (DT_NUM + DT_PROCNUM + DT_VERSIONTAGIDX (tag)) #define VERSTAG(tag) (DT_NUM + DT_PROCNUM + DT_VERSIONTAGIDX (tag))
@ -57,10 +58,10 @@ struct sym_val
something bad happened. */ something bad happened. */
static inline int static inline int
do_lookup (const char *undef_name, unsigned long int hash, do_lookup (const char *undef_name, unsigned long int hash,
const ElfW(Sym) **ref, struct sym_val *result, const ElfW(Sym) *ref, struct sym_val *result,
struct link_map *list[], size_t i, size_t n, struct link_map *list[], size_t i, size_t n,
const char *reference_name, const struct r_found_version *version, const char *reference_name, const struct r_found_version *version,
struct link_map *skip, int flags) struct link_map *skip, int reloc_type)
{ {
struct link_map *map; struct link_map *map;
@ -78,7 +79,8 @@ do_lookup (const char *undef_name, unsigned long int hash,
continue; continue;
/* Don't search the executable when resolving a copy reloc. */ /* Don't search the executable when resolving a copy reloc. */
if (flags & DL_LOOKUP_NOEXEC && map->l_type == lt_executable) if (elf_machine_lookup_noexec_p (reloc_type) &&
map->l_type == lt_executable)
continue; continue;
symtab = ((void *) map->l_addr + map->l_info[DT_SYMTAB]->d_un.d_ptr); symtab = ((void *) map->l_addr + map->l_info[DT_SYMTAB]->d_un.d_ptr);
@ -98,7 +100,7 @@ do_lookup (const char *undef_name, unsigned long int hash,
const ElfW(Sym) *sym = &symtab[symidx]; const ElfW(Sym) *sym = &symtab[symidx];
if (sym->st_value == 0 || /* No value. */ if (sym->st_value == 0 || /* No value. */
((flags & DL_LOOKUP_NOPLT) != 0 /* Reject PLT entry. */ (elf_machine_lookup_noplt_p (reloc_type) /* Reject PLT entry. */
&& sym->st_shndx == SHN_UNDEF)) && sym->st_shndx == SHN_UNDEF))
continue; continue;
@ -113,7 +115,7 @@ do_lookup (const char *undef_name, unsigned long int hash,
continue; continue;
} }
if (sym != *ref && strcmp (strtab + sym->st_name, undef_name)) if (sym != ref && strcmp (strtab + sym->st_name, undef_name))
/* Not the symbol we are looking for. */ /* Not the symbol we are looking for. */
continue; continue;
@ -188,16 +190,13 @@ do_lookup (const char *undef_name, unsigned long int hash,
} }
/* Search loaded objects' symbol tables for a definition of the symbol /* Search loaded objects' symbol tables for a definition of the symbol
UNDEF_NAME. FLAGS is a set of flags. If DL_LOOKUP_NOEXEC is set, UNDEF_NAME. */
then don't search the executable for a definition; this used for
copy relocs. If DL_LOOKUP_NOPLT is set, then a PLT entry cannot
satisfy the reference; some different binding must be found. */
ElfW(Addr) ElfW(Addr)
_dl_lookup_symbol (const char *undef_name, const ElfW(Sym) **ref, _dl_lookup_symbol (const char *undef_name, const ElfW(Sym) **ref,
struct link_map *symbol_scope[], struct link_map *symbol_scope[],
const char *reference_name, const char *reference_name,
int flags) int reloc_type)
{ {
const unsigned long int hash = _dl_elf_hash (undef_name); const unsigned long int hash = _dl_elf_hash (undef_name);
struct sym_val current_value = { 0, NULL }; struct sym_val current_value = { 0, NULL };
@ -205,9 +204,9 @@ _dl_lookup_symbol (const char *undef_name, const ElfW(Sym) **ref,
/* Search the relevant loaded objects for a definition. */ /* Search the relevant loaded objects for a definition. */
for (scope = symbol_scope; *scope; ++scope) for (scope = symbol_scope; *scope; ++scope)
if (do_lookup (undef_name, hash, ref, &current_value, if (do_lookup (undef_name, hash, *ref, &current_value,
(*scope)->l_searchlist, 0, (*scope)->l_nsearchlist, (*scope)->l_searchlist, 0, (*scope)->l_nsearchlist,
reference_name, NULL, NULL, flags)) reference_name, NULL, NULL, reloc_type))
break; break;
if (current_value.s == NULL && if (current_value.s == NULL &&
@ -230,8 +229,7 @@ ElfW(Addr)
_dl_lookup_symbol_skip (const char *undef_name, const ElfW(Sym) **ref, _dl_lookup_symbol_skip (const char *undef_name, const ElfW(Sym) **ref,
struct link_map *symbol_scope[], struct link_map *symbol_scope[],
const char *reference_name, const char *reference_name,
struct link_map *skip_map, struct link_map *skip_map)
int flags)
{ {
const unsigned long int hash = _dl_elf_hash (undef_name); const unsigned long int hash = _dl_elf_hash (undef_name);
struct sym_val current_value = { 0, NULL }; struct sym_val current_value = { 0, NULL };
@ -243,13 +241,13 @@ _dl_lookup_symbol_skip (const char *undef_name, const ElfW(Sym) **ref,
for (i = 0; (*scope)->l_dupsearchlist[i] != skip_map; ++i) for (i = 0; (*scope)->l_dupsearchlist[i] != skip_map; ++i)
assert (i < (*scope)->l_ndupsearchlist); assert (i < (*scope)->l_ndupsearchlist);
if (! do_lookup (undef_name, hash, ref, &current_value, if (! do_lookup (undef_name, hash, *ref, &current_value,
(*scope)->l_dupsearchlist, i, (*scope)->l_ndupsearchlist, (*scope)->l_dupsearchlist, i, (*scope)->l_ndupsearchlist,
reference_name, NULL, skip_map, flags)) reference_name, NULL, skip_map, 0))
while (*++scope) while (*++scope)
if (do_lookup (undef_name, hash, ref, &current_value, if (do_lookup (undef_name, hash, *ref, &current_value,
(*scope)->l_dupsearchlist, 0, (*scope)->l_ndupsearchlist, (*scope)->l_dupsearchlist, 0, (*scope)->l_ndupsearchlist,
reference_name, NULL, skip_map, flags)) reference_name, NULL, skip_map, 0))
break; break;
*ref = current_value.s; *ref = current_value.s;
@ -266,7 +264,8 @@ ElfW(Addr)
_dl_lookup_versioned_symbol (const char *undef_name, const ElfW(Sym) **ref, _dl_lookup_versioned_symbol (const char *undef_name, const ElfW(Sym) **ref,
struct link_map *symbol_scope[], struct link_map *symbol_scope[],
const char *reference_name, const char *reference_name,
const struct r_found_version *version, int flags) const struct r_found_version *version,
int reloc_type)
{ {
const unsigned long int hash = _dl_elf_hash (undef_name); const unsigned long int hash = _dl_elf_hash (undef_name);
struct sym_val current_value = { 0, NULL }; struct sym_val current_value = { 0, NULL };
@ -275,9 +274,9 @@ _dl_lookup_versioned_symbol (const char *undef_name, const ElfW(Sym) **ref,
/* Search the relevant loaded objects for a definition. */ /* Search the relevant loaded objects for a definition. */
for (scope = symbol_scope; *scope; ++scope) for (scope = symbol_scope; *scope; ++scope)
{ {
int res = do_lookup (undef_name, hash, ref, &current_value, int res = do_lookup (undef_name, hash, *ref, &current_value,
(*scope)->l_searchlist, 0, (*scope)->l_nsearchlist, (*scope)->l_searchlist, 0, (*scope)->l_nsearchlist,
reference_name, version, NULL, flags); reference_name, version, NULL, reloc_type);
if (res > 0) if (res > 0)
break; break;
@ -312,8 +311,7 @@ _dl_lookup_versioned_symbol_skip (const char *undef_name,
struct link_map *symbol_scope[], struct link_map *symbol_scope[],
const char *reference_name, const char *reference_name,
const struct r_found_version *version, const struct r_found_version *version,
struct link_map *skip_map, struct link_map *skip_map)
int flags)
{ {
const unsigned long int hash = _dl_elf_hash (undef_name); const unsigned long int hash = _dl_elf_hash (undef_name);
struct sym_val current_value = { 0, NULL }; struct sym_val current_value = { 0, NULL };
@ -325,13 +323,13 @@ _dl_lookup_versioned_symbol_skip (const char *undef_name,
for (i = 0; (*scope)->l_dupsearchlist[i] != skip_map; ++i) for (i = 0; (*scope)->l_dupsearchlist[i] != skip_map; ++i)
assert (i < (*scope)->l_ndupsearchlist); assert (i < (*scope)->l_ndupsearchlist);
if (! do_lookup (undef_name, hash, ref, &current_value, if (! do_lookup (undef_name, hash, *ref, &current_value,
(*scope)->l_dupsearchlist, i, (*scope)->l_ndupsearchlist, (*scope)->l_dupsearchlist, i, (*scope)->l_ndupsearchlist,
reference_name, version, skip_map, flags)) reference_name, version, skip_map, 0))
while (*++scope) while (*++scope)
if (do_lookup (undef_name, hash, ref, &current_value, if (do_lookup (undef_name, hash, *ref, &current_value,
(*scope)->l_dupsearchlist, 0, (*scope)->l_ndupsearchlist, (*scope)->l_dupsearchlist, 0, (*scope)->l_ndupsearchlist,
reference_name, version, skip_map, flags)) reference_name, version, skip_map, 0))
break; break;
*ref = current_value.s; *ref = current_value.s;

View File

@ -58,10 +58,7 @@ _dl_relocate_object (struct link_map *l, struct link_map *scope[], int lazy)
/* This macro is used as a callback from the ELF_DYNAMIC_RELOCATE code. */ /* This macro is used as a callback from the ELF_DYNAMIC_RELOCATE code. */
#define RESOLVE(ref, version, flags) \ #define RESOLVE(ref, version, flags) \
((*ref)->st_shndx != SHN_UNDEF && \ ((version) != NULL && (version)->hash != 0 \
ELFW(ST_BIND) ((*ref)->st_info) == STB_LOCAL \
? l->l_addr : \
(version) != NULL && (version)->hash != 0 \
? _dl_lookup_versioned_symbol (strtab + (*ref)->st_name, (ref), scope, \ ? _dl_lookup_versioned_symbol (strtab + (*ref)->st_name, (ref), scope, \
l->l_name, (version), (flags)) \ l->l_name, (version), (flags)) \
: _dl_lookup_symbol (strtab + (*ref)->st_name, (ref), scope, \ : _dl_lookup_symbol (strtab + (*ref)->st_name, (ref), scope, \

View File

@ -1,5 +1,5 @@
/* Support for dynamic linking code in static libc. /* Support for dynamic linking code in static libc.
Copyright (C) 1996 Free Software Foundation, Inc. Copyright (C) 1996, 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library. This file is part of the GNU C Library.
The GNU C Library is free software; you can redistribute it and/or The GNU C Library is free software; you can redistribute it and/or
@ -62,7 +62,7 @@ _dl_sysdep_read_whole_file (const char *file, size_t *sizep, int prot)
int fd = __open (file, O_RDONLY); int fd = __open (file, O_RDONLY);
if (fd < 0) if (fd < 0)
return NULL; return NULL;
if (__fstat (fd, &st) < 0) if (__fxstat (_STAT_VER, fd, &st) < 0)
result = NULL; result = NULL;
else else
{ {

View File

@ -53,8 +53,7 @@ RTLD_NEXT used in code not dynamically loaded"));
while (l->l_loader) while (l->l_loader)
l = l->l_loader; l = l->l_loader;
loadbase = _dl_lookup_symbol_skip loadbase = _dl_lookup_symbol_skip (name, &ref, &_dl_loaded, NULL, l);
(name, &ref, &_dl_loaded, NULL, l, 0);
} }
else else
{ {

View File

@ -57,7 +57,7 @@ RTLD_NEXT used in code not dynamically loaded"));
l = l->l_loader; l = l->l_loader;
loadbase = _dl_lookup_versioned_symbol_skip loadbase = _dl_lookup_versioned_symbol_skip
(name, &ref, &_dl_loaded, NULL, &version, l, 0); (name, &ref, &_dl_loaded, NULL, &version, l);
} }
else else
{ {

View File

@ -290,18 +290,15 @@ extern void _dl_close (struct link_map *map);
null-terminated list of object scopes to search; each object's null-terminated list of object scopes to search; each object's
l_searchlist (i.e. the segment of the dependency tree starting at that l_searchlist (i.e. the segment of the dependency tree starting at that
object) is searched in turn. REFERENCE_NAME should name the object object) is searched in turn. REFERENCE_NAME should name the object
containing the reference; it is used in error messages. FLAGS is a containing the reference; it is used in error messages.
set of flags: */ RELOC_TYPE is a machine-dependent reloc type, which is passed to
#define DL_LOOKUP_NOEXEC 1 /* Don't search the executable for a the `elf_machine_lookup_*_p' macros in dl-machine.h to affect which
definition; this is used for copy symbols can be chosen. */
relocs. */
#define DL_LOOKUP_NOPLT 2 /* The reference must not be resolved
to a PLT entry. */
extern ElfW(Addr) _dl_lookup_symbol (const char *undef, extern ElfW(Addr) _dl_lookup_symbol (const char *undef,
const ElfW(Sym) **sym, const ElfW(Sym) **sym,
struct link_map *symbol_scope[], struct link_map *symbol_scope[],
const char *reference_name, const char *reference_name,
int flags); int reloc_type);
/* Lookup versioned symbol. */ /* Lookup versioned symbol. */
extern ElfW(Addr) _dl_lookup_versioned_symbol (const char *undef, extern ElfW(Addr) _dl_lookup_versioned_symbol (const char *undef,
@ -309,15 +306,14 @@ extern ElfW(Addr) _dl_lookup_versioned_symbol (const char *undef,
struct link_map *symbol_scope[], struct link_map *symbol_scope[],
const char *reference_name, const char *reference_name,
const struct r_found_version *version, const struct r_found_version *version,
int flags); int reloc_type);
/* For handling RTLD_NEXT we must be able to skip shared objects. */ /* For handling RTLD_NEXT we must be able to skip shared objects. */
extern ElfW(Addr) _dl_lookup_symbol_skip (const char *undef, extern ElfW(Addr) _dl_lookup_symbol_skip (const char *undef,
const ElfW(Sym) **sym, const ElfW(Sym) **sym,
struct link_map *symbol_scope[], struct link_map *symbol_scope[],
const char *reference_name, const char *reference_name,
struct link_map *skip_this, struct link_map *skip_this);
int flags);
/* For handling RTLD_NEXT with versioned symbols we must be able to /* For handling RTLD_NEXT with versioned symbols we must be able to
skip shared objects. */ skip shared objects. */
@ -326,8 +322,7 @@ extern ElfW(Addr) _dl_lookup_versioned_symbol_skip (const char *undef,
struct link_map *symbol_scope[], struct link_map *symbol_scope[],
const char *reference_name, const char *reference_name,
const struct r_found_version *version, const struct r_found_version *version,
struct link_map *skip_this, struct link_map *skip_this);
int flags);
/* Look up symbol NAME in MAP's scope and return its run-time address. */ /* Look up symbol NAME in MAP's scope and return its run-time address. */
extern ElfW(Addr) _dl_symbol_value (struct link_map *map, const char *name); extern ElfW(Addr) _dl_symbol_value (struct link_map *map, const char *name);

View File

@ -518,7 +518,7 @@ of this helper program; chances are you did not intend to run this program.\n",
ElfW(Addr) loadbase = _dl_lookup_symbol (_dl_argv[i], &ref, ElfW(Addr) loadbase = _dl_lookup_symbol (_dl_argv[i], &ref,
&_dl_default_scope[2], &_dl_default_scope[2],
"argument", "argument",
DL_LOOKUP_NOPLT); ELF_MACHINE_RELOC_NOPLT);
char buf[20], *bp; char buf[20], *bp;
buf[sizeof buf - 1] = '\0'; buf[sizeof buf - 1] = '\0';
bp = _itoa (ref->st_value, &buf[sizeof buf - 1], 16, 0); bp = _itoa (ref->st_value, &buf[sizeof buf - 1], 16, 0);

View File

@ -47,7 +47,8 @@ libm-calls = e_acos e_acosh e_asin e_atan2 e_atanh e_cosh e_exp e_fmod \
w_atan2 w_atanh w_cosh w_drem w_exp w_fmod w_gamma \ w_atan2 w_atanh w_cosh w_drem w_exp w_fmod w_gamma \
w_gamma_r w_hypot w_j0 w_j1 w_jn w_lgamma w_lgamma_r \ w_gamma_r w_hypot w_j0 w_j1 w_jn w_lgamma w_lgamma_r \
w_log w_log10 w_pow w_remainder w_scalb w_sinh w_sqrt \ w_log w_log10 w_pow w_remainder w_scalb w_sinh w_sqrt \
s_signbit s_fpclassify s_fmax s_fmin s_fdim \ s_signbit s_fpclassify s_fmax s_fmin s_fdim s_nan s_trunc \
s_remquo \
conj cimag creal cabs conj cimag creal cabs
libm-routines = $(libm-support) $(libm-calls) \ libm-routines = $(libm-support) $(libm-calls) \
$(patsubst %_rf,%f_r,$(libm-calls:=f)) \ $(patsubst %_rf,%f_r,$(libm-calls:=f)) \

View File

@ -102,10 +102,10 @@ static MATHTYPE plus_infty, minus_infty, nan_value;
typedef MATHTYPE (*mathfunc) (MATHTYPE); typedef MATHTYPE (*mathfunc) (MATHTYPE);
#define ISINF(x) \ #define ISINF(x) \
(sizeof (x) == sizeof (float) ? \ (sizeof (x) == sizeof (float) ? \
isinff (x) \ isinff (x) \
: sizeof (x) == sizeof (double) ? \ : sizeof (x) == sizeof (double) ? \
isinf (x) : isinfl (x)) isinf (x) : isinfl (x))
@ -1222,6 +1222,222 @@ pow_test (void)
} }
static void
fdim_test (void)
{
check ("fdim (+0, +0) = +0", FUNC(fdim) (0, 0), 0);
check ("fdim (9, 0) = 9", FUNC(fdim) (9, 0), 9);
check ("fdim (0, 9) = 0", FUNC(fdim) (0, 9), 0);
check ("fdim (-9, 0) = 9", FUNC(fdim) (-9, 0), 0);
check ("fdim (0, -9) = 9", FUNC(fdim) (0, -9), 9);
check_isinfp ("fdim (+inf, 9) = +inf", FUNC(fdim) (plus_infty, 9));
check_isinfp ("fdim (+inf, -9) = +inf", FUNC(fdim) (plus_infty, -9));
check ("fdim (-inf, 9) = 0", FUNC(fdim) (minus_infty, 9), 0);
check ("fdim (-inf, -9) = 0", FUNC(fdim) (minus_infty, -9), 0);
check_isinfp ("fdim (+9, -inf) = +inf", FUNC(fdim) (9, minus_infty));
check_isinfp ("fdim (-9, -inf) = +inf", FUNC(fdim) (-9, minus_infty));
check ("fdim (9, inf) = 0", FUNC(fdim) (9, plus_infty), 0);
check ("fdim (-9, inf) = 0", FUNC(fdim) (-9, plus_infty), 0);
check_isnan ("fdim (0, NaN) = NaN", FUNC(fdim) (0, nan_value));
check_isnan ("fdim (9, NaN) = NaN", FUNC(fdim) (9, nan_value));
check_isnan ("fdim (-9, NaN) = NaN", FUNC(fdim) (-9, nan_value));
check_isnan ("fdim (NaN, 9) = NaN", FUNC(fdim) (nan_value, 9));
check_isnan ("fdim (NaN, -9) = NaN", FUNC(fdim) (nan_value, -9));
check_isnan ("fdim (+inf, NaN) = NaN", FUNC(fdim) (plus_infty, nan_value));
check_isnan ("fdim (-inf, NaN) = NaN", FUNC(fdim) (minus_infty, nan_value));
check_isnan ("fdim (NaN, +inf) = NaN", FUNC(fdim) (nan_value, plus_infty));
check_isnan ("fdim (NaN, -inf) = NaN", FUNC(fdim) (nan_value, minus_infty));
check_isnan ("fdim (NaN, NaN) = NaN", FUNC(fdim) (nan_value, nan_value));
}
static void
fmin_test (void)
{
check ("fmin (+0, +0) = +0", FUNC(fmin) (0, 0), 0);
check ("fmin (9, 0) = 0", FUNC(fmin) (9, 0), 0);
check ("fmin (0, 9) = 0", FUNC(fmin) (0, 9), 0);
check ("fmin (-9, 0) = -9", FUNC(fmin) (-9, 0), -9);
check ("fmin (0, -9) = -9", FUNC(fmin) (0, -9), -9);
check ("fmin (+inf, 9) = 9", FUNC(fmin) (plus_infty, 9), 9);
check ("fmin (9, +inf) = 9", FUNC(fmin) (9, plus_infty), 9);
check ("fmin (+inf, -9) = -9", FUNC(fmin) (plus_infty, -9), -9);
check ("fmin (-9, +inf) = -9", FUNC(fmin) (-9, plus_infty), -9);
check_isinfn ("fmin (-inf, 9) = -inf", FUNC(fmin) (minus_infty, 9));
check_isinfn ("fmin (-inf, -9) = -inf", FUNC(fmin) (minus_infty, -9));
check_isinfn ("fmin (+9, -inf) = -inf", FUNC(fmin) (9, minus_infty));
check_isinfn ("fmin (-9, -inf) = -inf", FUNC(fmin) (-9, minus_infty));
check ("fmin (0, NaN) = 0", FUNC(fmin) (0, nan_value), 0);
check ("fmin (9, NaN) = 9", FUNC(fmin) (9, nan_value), 9);
check ("fmin (-9, NaN) = 9", FUNC(fmin) (-9, nan_value), -9);
check ("fmin (NaN, 0) = 0", FUNC(fmin) (nan_value, 0), 0);
check ("fmin (NaN, 9) = NaN", FUNC(fmin) (nan_value, 9), 9);
check ("fmin (NaN, -9) = NaN", FUNC(fmin) (nan_value, -9), -9);
check_isinfp ("fmin (+inf, NaN) = +inf", FUNC(fmin) (plus_infty, nan_value));
check_isinfn ("fmin (-inf, NaN) = -inf", FUNC(fmin) (minus_infty, nan_value));
check_isinfp ("fmin (NaN, +inf) = +inf", FUNC(fmin) (nan_value, plus_infty));
check_isinfn ("fmin (NaN, -inf) = -inf", FUNC(fmin) (nan_value, minus_infty));
check_isnan ("fmin (NaN, NaN) = NaN", FUNC(fmin) (nan_value, nan_value));
}
static void
fmax_test (void)
{
check ("fmax (+0, +0) = +0", FUNC(fmax) (0, 0), 0);
check ("fmax (9, 0) = 9", FUNC(fmax) (9, 0), 9);
check ("fmax (0, 9) = 9", FUNC(fmax) (0, 9), 9);
check ("fmax (-9, 0) = 0", FUNC(fmax) (-9, 0), 0);
check ("fmax (0, -9) = 0", FUNC(fmax) (0, -9), 0);
check_isinfp ("fmax (+inf, 9) = +inf", FUNC(fmax) (plus_infty, 9));
check_isinfp ("fmax (9, +inf) = +inf", FUNC(fmax) (0, plus_infty));
check_isinfp ("fmax (-9, +inf) = +inf", FUNC(fmax) (-9, plus_infty));
check_isinfp ("fmax (+inf, -9) = +inf", FUNC(fmax) (plus_infty, -9));
check ("fmax (-inf, 9) = 9", FUNC(fmax) (minus_infty, 9), 9);
check ("fmax (-inf, -9) = -9", FUNC(fmax) (minus_infty, -9), -9);
check ("fmax (+9, -inf) = 9", FUNC(fmax) (9, minus_infty), 9);
check ("fmax (-9, -inf) = -9", FUNC(fmax) (-9, minus_infty), -9);
check ("fmax (0, NaN) = 0", FUNC(fmax) (0, nan_value), 0);
check ("fmax (9, NaN) = 9", FUNC(fmax) (9, nan_value), 9);
check ("fmax (-9, NaN) = 9", FUNC(fmax) (-9, nan_value), -9);
check ("fmax (NaN, 0) = 0", FUNC(fmax) (nan_value, 0), 0);
check ("fmax (NaN, 9) = NaN", FUNC(fmax) (nan_value, 9), 9);
check ("fmax (NaN, -9) = NaN", FUNC(fmax) (nan_value, -9), -9);
check_isinfp ("fmax (+inf, NaN) = +inf", FUNC(fmax) (plus_infty, nan_value));
check_isinfn ("fmax (-inf, NaN) = -inf", FUNC(fmax) (minus_infty, nan_value));
check_isinfp ("fmax (NaN, +inf) = +inf", FUNC(fmax) (nan_value, plus_infty));
check_isinfn ("fmax (NaN, -inf) = -inf", FUNC(fmax) (nan_value, minus_infty));
check_isnan ("fmax (NaN, NaN) = NaN", FUNC(fmax) (nan_value, nan_value));
}
static void
nextafter_test (void)
{
MATHTYPE x;
check ("nextafter (+0, +0) = +0", FUNC(nextafter) (0, 0), 0);
check ("nextafter (-0, +0) = +0", FUNC(nextafter) (minus_zero, 0), 0);
check ("nextafter (+0, -0) = -0", FUNC(nextafter) (0, minus_zero),
minus_zero);
check ("nextafter (-0, -0) = -0", FUNC(nextafter) (minus_zero, minus_zero),
minus_zero);
check ("nextafter (9, 9) = 9", FUNC(nextafter) (9, 9), 9);
check ("nextafter (-9, -9) = -9", FUNC(nextafter) (-9, -9), -9);
check_isinfp ("nextafter (+inf, +inf) = +inf",
FUNC(nextafter) (plus_infty, plus_infty));
check_isinfn ("nextafter (-inf, -inf) = -inf",
FUNC(nextafter) (minus_infty, minus_infty));
x = rand () * 1.1;
check_isnan ("nextafter (NaN, x) = NaN", FUNC(nextafter) (nan_value, x));
check_isnan ("nextafter (x, NaN) = NaN", FUNC(nextafter) (x, nan_value));
check_isnan ("nextafter (NaN, NaN) = NaN", FUNC(nextafter) (nan_value,
nan_value));
/* XXX We need the hexadecimal FP number representation here for further
tests. */
}
static void
copysign_test (void)
{
check ("copysign (0, 4) = 0", FUNC(copysign) (0, 4), 0);
check ("copysign (0, -4) = -0", FUNC(copysign) (0, -4), minus_zero);
check ("copysign (-0, 4) = 0", FUNC(copysign) (minus_zero, 4), 0);
check ("copysign (-0, -4) = -0", FUNC(copysign) (minus_zero, -4),
minus_zero);
check_isinfp ("copysign (+inf, 0) = +inf", FUNC(copysign) (plus_infty, 0));
check_isinfn ("copysign (+inf, -0) = -inf", FUNC(copysign) (plus_infty,
minus_zero));
check_isinfp ("copysign (-inf, 0) = +inf", FUNC(copysign) (minus_infty, 0));
check_isinfn ("copysign (-inf, -0) = -inf", FUNC(copysign) (minus_infty,
minus_zero));
check ("copysign (0, +inf) = 0", FUNC(copysign) (0, plus_infty), 0);
check ("copysign (0, -inf) = -0", FUNC(copysign) (0, minus_zero),
minus_zero);
check ("copysign (-0, +inf) = 0", FUNC(copysign) (minus_zero, plus_infty),
0);
check ("copysign (-0, -inf) = -0", FUNC(copysign) (minus_zero, minus_zero),
minus_zero);
/* XXX More correctly we would have to check the sign of the NaN. */
check_isnan ("copysign (+NaN, 0) = +inf", FUNC(copysign) (nan_value, 0));
check_isnan ("copysign (+NaN, -0) = -inf", FUNC(copysign) (nan_value,
minus_zero));
check_isnan ("copysign (-NaN, 0) = +inf", FUNC(copysign) (-nan_value, 0));
check_isnan ("copysign (-NaN, -0) = -inf", FUNC(copysign) (-nan_value,
minus_zero));
}
static void
trunc_test (void)
{
check ("trunc(0) = 0", FUNC(trunc) (0), 0);
check ("trunc(-0) = -0", FUNC(trunc) (minus_zero), minus_zero);
check ("trunc(0.625) = 0", FUNC(trunc) (0.625), 0);
check ("trunc(-0.625) = -0", FUNC(trunc) (-0.625), minus_zero);
check ("trunc(1) = 1", FUNC(trunc) (1), 1);
check ("trunc(-1) = -1", FUNC(trunc) (-1), -1);
check ("trunc(1.625) = 1", FUNC(trunc) (1.625), 1);
check ("trunc(-1.625) = -1", FUNC(trunc) (-1.625), -1);
check ("trunc(1048580.625) = 1048580", FUNC(trunc) (1048580.625L),
1048580L);
check ("trunc(-1048580.625) = -1048580", FUNC(trunc) (-1048580.625L),
-1048580L);
check ("trunc(8388610.125) = 8388610", FUNC(trunc) (8388610.125L),
8388610.0L);
check ("trunc(-8388610.125) = -8388610", FUNC(trunc) (-8388610.125L),
-8388610.0L);
check ("trunc(4294967296.625) = 4294967296", FUNC(trunc) (4294967296.625L),
4294967296.0L);
check ("trunc(-4294967296.625) = -4294967296",
FUNC(trunc) (-4294967296.625L), -4294967296.0L);
check_isinfp ("trunc(+inf) = +inf", FUNC(trunc) (plus_infty));
check_isinfn ("trunc(-inf) = -inf", FUNC(trunc) (minus_infty));
check_isnan ("trunc(NaN) = NaN", FUNC(trunc) (nan_value));
}
static void
remquo_test (void)
{
int quo;
MATHTYPE result;
result = FUNC(remquo) (1.625, 1.0, &quo);
check ("remquo(1.625, 1.0, &x) == -0.375", result, -0.375);
check ("remquo(1.625, 1.0, &x) puts 1 in x", quo, 1);
result = FUNC(remquo) (-1.625, 1.0, &quo);
check ("remquo(-1.625, 1.0, &x) == 0.375", result, 0.375);
check ("remquo(-1.625, 1.0, &x) puts 1 in x", quo, -1);
result = FUNC(remquo) (1.625, -1.0, &quo);
check ("remquo(1.125, -1.0, &x) == 0.125", result, 0.125);
check ("remquo(1.125, -1.0, &x) puts 1 in x", quo, -1);
result = FUNC(remquo) (-1.625, -1.0, &quo);
check ("remquo(-1.125, -1.0, &x) == 0.125", result, 0.125);
check ("remquo(-1.125, -1.0, &x) puts 1 in x", quo, 1);
}
static void static void
inverse_func_pair_test (const char *test_name, inverse_func_pair_test (const char *test_name,
mathfunc f1, mathfunc inverse, mathfunc f1, mathfunc inverse,
@ -1244,33 +1460,33 @@ inverse_func_pair_test (const char *test_name,
static void static void
inverse_functions (void) inverse_functions (void)
{ {
inverse_func_pair_test ("(asin(sin(x)) == x", inverse_func_pair_test ("asin(sin(x)) == x",
FUNC(sin), FUNC(asin), 1.0, CHOOSE (0, 0, 1e-7L)); FUNC(sin), FUNC(asin), 1.0, CHOOSE (0, 0, 1e-7L));
inverse_func_pair_test ("(sin(asin(x)) == x", inverse_func_pair_test ("sin(asin(x)) == x",
FUNC(asin), FUNC(sin), 1.0, 0.0); FUNC(asin), FUNC(sin), 1.0, 0.0);
inverse_func_pair_test ("(acos(cos(x)) == x", inverse_func_pair_test ("acos(cos(x)) == x",
FUNC(cos), FUNC(acos), 1.0, CHOOSE (0, 1e-15L, 0)); FUNC(cos), FUNC(acos), 1.0, CHOOSE (0, 1e-15L, 0));
inverse_func_pair_test ("(cos(acos(x)) == x", inverse_func_pair_test ("cos(acos(x)) == x",
FUNC(acos), FUNC(cos), 1.0, 0.0); FUNC(acos), FUNC(cos), 1.0, 0.0);
inverse_func_pair_test ("(atan(tan(x)) == x", inverse_func_pair_test ("atan(tan(x)) == x",
FUNC(tan), FUNC(atan), 1.0, 0.0); FUNC(tan), FUNC(atan), 1.0, 0.0);
inverse_func_pair_test ("(tan(atan(x)) == x", inverse_func_pair_test ("tan(atan(x)) == x",
FUNC(atan), FUNC(tan), 1.0, CHOOSE (0, 1e-15L, 0)); FUNC(atan), FUNC(tan), 1.0, CHOOSE (0, 1e-15L, 0));
inverse_func_pair_test ("(asinh(sinh(x)) == x", inverse_func_pair_test ("asinh(sinh(x)) == x",
FUNC(sinh), FUNC(asinh), 1.0, CHOOSE (1e-18L, 0, 0)); FUNC(sinh), FUNC(asinh), 1.0, CHOOSE (1e-18L, 0, 0));
inverse_func_pair_test ("(sinh(asinh(x)) == x", inverse_func_pair_test ("sinh(asinh(x)) == x",
FUNC(asinh), FUNC(sinh), 1.0, 0.0); FUNC(asinh), FUNC(sinh), 1.0, 0.0);
inverse_func_pair_test ("(acosh(cosh(x)) == x", inverse_func_pair_test ("acosh(cosh(x)) == x",
FUNC(cosh), FUNC(acosh), 1.0, CHOOSE (1e-18L, 1e-15L, 0)); FUNC(cosh), FUNC(acosh), 1.0, CHOOSE (1e-18L, 1e-15L, 0));
inverse_func_pair_test ("(cosh(acosh(x)) == x", inverse_func_pair_test ("cosh(acosh(x)) == x",
FUNC(acosh), FUNC(cosh), 1.0, 0.0); FUNC(acosh), FUNC(cosh), 1.0, 0.0);
inverse_func_pair_test ("(atanh(tanh(x)) == x", inverse_func_pair_test ("atanh(tanh(x)) == x",
FUNC(tanh), FUNC(atanh), 1.0, CHOOSE (0, 1e-15L, 0)); FUNC(tanh), FUNC(atanh), 1.0, CHOOSE (0, 1e-15L, 0));
inverse_func_pair_test ("(tanh(atanh(x)) == x", inverse_func_pair_test ("tanh(atanh(x)) == x",
FUNC(atanh), FUNC(tanh), 1.0, 0.0); FUNC(atanh), FUNC(tanh), 1.0, 0.0);
} }
@ -1401,6 +1617,15 @@ basic_tests (void)
check_bool ("!isinf (-NAN)", !(isinf (-NAN))); check_bool ("!isinf (-NAN)", !(isinf (-NAN)));
check_bool ("NAN != NAN", NAN != NAN); check_bool ("NAN != NAN", NAN != NAN);
/*
And again with the value returned by the `nan' function.
*/
check_bool ("isnan (NAN)", isnan (FUNC(nan) ("")));
check_bool ("isnan (-NAN)", isnan (-FUNC(nan) ("")));
check_bool ("!isinf (NAN)", !(isinf (FUNC(nan) (""))));
check_bool ("!isinf (-NAN)", !(isinf (-FUNC(nan) (""))));
check_bool ("NAN != NAN", FUNC(nan) ("") != FUNC(nan) (""));
/* test if EPSILON is ok */ /* test if EPSILON is ok */
x1 = MATHCONST (1.0); x1 = MATHCONST (1.0);
x2 = x1 + CHOOSE (LDBL_EPSILON, DBL_EPSILON, FLT_EPSILON); x2 = x1 + CHOOSE (LDBL_EPSILON, DBL_EPSILON, FLT_EPSILON);
@ -1520,6 +1745,16 @@ main (int argc, char *argv[])
fpclassify_test (); fpclassify_test ();
hypot_test (); hypot_test ();
pow_test (); pow_test ();
fdim_test ();
fmin_test ();
fmax_test ();
nextafter_test ();
copysign_test ();
trunc_test ();
#if 0
/* XXX I'm not sure what is the correct result. */
remquo_test ();
#endif
identities (); identities ();
inverse_functions (); inverse_functions ();

View File

@ -154,7 +154,7 @@ _dl_sysdep_read_whole_file (const char *file, size_t *sizep, int prot)
int fd = __open (file, O_RDONLY); int fd = __open (file, O_RDONLY);
if (fd < 0) if (fd < 0)
return NULL; return NULL;
if (__fstat (fd, &st) < 0) if (__fxstat (_STAT_VER, fd, &st) < 0)
result = NULL; result = NULL;
else else
{ {

View File

@ -218,7 +218,14 @@ _dl_start_user:\n\
/* Nonzero iff TYPE describes relocation of a PLT entry, so /* Nonzero iff TYPE describes relocation of a PLT entry, so
PLT entries should not be allowed to define the value. */ PLT entries should not be allowed to define the value. */
#define elf_machine_pltrel_p(type) ((type) == R_386_JMP_SLOT) #define elf_machine_lookup_noexec_p(type) ((type) == R_386_COPY)
/* Nonzero iff TYPE describes relocation of a PLT entry, so
PLT entries should not be allowed to define the value. */
#define elf_machine_lookup_noplt_p(type) ((type) == R_386_JMP_SLOT)
/* A reloc type used for ld.so cmdline arg lookups to reject PLT entries. */
#define ELF_MACHINE_RELOC_NOPLT R_386_JMP_SLOT
/* The i386 never uses Elf32_Rela relocations. */ /* The i386 never uses Elf32_Rela relocations. */
#define ELF_MACHINE_NO_RELA 1 #define ELF_MACHINE_NO_RELA 1
@ -235,62 +242,63 @@ elf_machine_rel (struct link_map *map, const Elf32_Rel *reloc,
const Elf32_Sym *sym, const struct r_found_version *version) const Elf32_Sym *sym, const struct r_found_version *version)
{ {
Elf32_Addr *const reloc_addr = (void *) (map->l_addr + reloc->r_offset); Elf32_Addr *const reloc_addr = (void *) (map->l_addr + reloc->r_offset);
Elf32_Addr loadbase;
switch (ELF32_R_TYPE (reloc->r_info)) if (ELF32_R_TYPE (reloc->r_info) == R_386_RELATIVE)
{ {
case R_386_COPY:
loadbase = RESOLVE (&sym, version, DL_LOOKUP_NOEXEC);
memcpy (reloc_addr, (void *) (loadbase + sym->st_value), sym->st_size);
break;
case R_386_GLOB_DAT:
loadbase = RESOLVE (&sym, version, 0);
*reloc_addr = sym ? (loadbase + sym->st_value) : 0;
break;
case R_386_JMP_SLOT:
loadbase = RESOLVE (&sym, version, DL_LOOKUP_NOPLT);
*reloc_addr = sym ? (loadbase + sym->st_value) : 0;
break;
case R_386_32:
{
Elf32_Addr undo = 0;
#ifndef RTLD_BOOTSTRAP
/* This is defined in rtld.c, but nowhere in the static libc.a;
make the reference weak so static programs can still link. This
declaration cannot be done when compiling rtld.c (i.e. #ifdef
RTLD_BOOTSTRAP) because rtld.c contains the common defn for
_dl_rtld_map, which is incompatible with a weak decl in the same
file. */
weak_extern (_dl_rtld_map);
if (map == &_dl_rtld_map)
/* Undo the relocation done here during bootstrapping. Now we will
relocate it anew, possibly using a binding found in the user
program or a loaded library rather than the dynamic linker's
built-in definitions used while loading those libraries. */
undo = map->l_addr + sym->st_value;
#endif
loadbase = RESOLVE (&sym, version, 0);
*reloc_addr += (sym ? (loadbase + sym->st_value) : 0) - undo;
break;
}
case R_386_RELATIVE:
#ifndef RTLD_BOOTSTRAP #ifndef RTLD_BOOTSTRAP
if (map != &_dl_rtld_map) /* Already done in rtld itself. */ if (map != &_dl_rtld_map) /* Already done in rtld itself. */
#endif #endif
*reloc_addr += map->l_addr; *reloc_addr += map->l_addr;
break;
case R_386_PC32:
loadbase = RESOLVE (&sym, version, 0);
*reloc_addr += ((sym ? (loadbase + sym->st_value) : 0) -
(Elf32_Addr) reloc_addr);
break;
case R_386_NONE: /* Alright, Wilbur. */
break;
default:
assert (! "unexpected dynamic reloc type");
break;
} }
else
{
#ifndef RTLD_BOOTSTRAP
const Elf32_Sym *const refsym = sym;
#endif
Elf32_Addr value = RESOLVE (&sym, version, ELF32_R_TYPE (reloc->r_info));
if (sym)
value += sym->st_value;
switch (ELF32_R_TYPE (reloc->r_info))
{
case R_386_COPY:
memcpy (reloc_addr, (void *) value, sym->st_size);
break;
case R_386_GLOB_DAT:
case R_386_JMP_SLOT:
*reloc_addr = value;
break;
case R_386_32:
{
#ifndef RTLD_BOOTSTRAP
/* This is defined in rtld.c, but nowhere in the static
libc.a; make the reference weak so static programs can
still link. This declaration cannot be done when
compiling rtld.c (i.e. #ifdef RTLD_BOOTSTRAP) because
rtld.c contains the common defn for _dl_rtld_map, which
is incompatible with a weak decl in the same file. */
weak_extern (_dl_rtld_map);
if (map == &_dl_rtld_map)
/* Undo the relocation done here during bootstrapping.
Now we will relocate it anew, possibly using a
binding found in the user program or a loaded library
rather than the dynamic linker's built-in definitions
used while loading those libraries. */
value -= map->l_addr + refsym->st_value;
#endif
*reloc_addr += value;
break;
}
case R_386_PC32:
*reloc_addr += (value - (Elf32_Addr) reloc_addr);
break;
case R_386_NONE: /* Alright, Wilbur. */
break;
default:
assert (! "unexpected dynamic reloc type");
break;
}
}
} }
static inline void static inline void

View File

@ -49,7 +49,7 @@ static char rcsid[] = "$NetBSD: $";
if(((ix==0x7fff)&&(((hx|lx)|-(hx|lx))&hx)>>31!=0) || /* x is nan */ if(((ix==0x7fff)&&(((hx|lx)|-(hx|lx))&hx)>>31!=0) || /* x is nan */
((iy==0x7fff)&&(((hy|ly)|-(hy|ly))&hy)>>31!=0)) /* y is nan */ ((iy==0x7fff)&&(((hy|ly)|-(hy|ly))&hy)>>31!=0)) /* y is nan */
return x+y; return x+y;
if(x==y) return x; /* x=y, return x */ if(x==y) return y; /* x=y, return y */
if((ix|hx|lx)==0) { /* x == 0 */ if((ix|hx|lx)==0) { /* x == 0 */
SET_LDOUBLE_WORDS(x,esx&0x8000,0,1);/* return +-minsubnormal */ SET_LDOUBLE_WORDS(x,esx&0x8000,0,1);/* return +-minsubnormal */
y = x*x; y = x*x;

View File

@ -0,0 +1,31 @@
/*
* Written by Ulrich Drepper <drepper@cygnus.com>.
* Based on e_remainder by J.T. Conklin <jtc@netbsd.org>.
* Public domain.
*/
#include <machine/asm.h>
ENTRY(__remquo)
fldl 12(%esp)
fldl 4(%esp)
1: fprem1
fstsw %ax
sahf
jp 1b
fstpl %st(1)
/* Compute the congruent of the quotient. */
movl %eax, %ecx
shrl $8, %eax
shrl $12, %ecx
andl $4, %ecx
andl $3, %eax
orl %eax, %ecx
movl $0xef2960, %eax
shrl %cl, %eax
andl $3, %eax
movl 20(%esp), %ecx
movl %eax, (%ecx)
ret
END (__remquo)
weak_alias (__remquo, remquo)

View File

@ -0,0 +1,31 @@
/*
* Written by Ulrich Drepper <drepper@cygnus.com>.
* Based on e_remainder by J.T. Conklin <jtc@netbsd.org>.
* Public domain.
*/
#include <machine/asm.h>
ENTRY(__remquof)
flds 8(%esp)
flds 4(%esp)
1: fprem1
fstsw %ax
sahf
jp 1b
fstpl %st(1)
/* Compute the congruent of the quotient. */
movl %eax, %ecx
shrl $8, %eax
shrl $12, %ecx
andl $4, %ecx
andl $3, %eax
orl %eax, %ecx
movl $0xef2960, %eax
shrl %cl, %eax
andl $3, %eax
movl 12(%esp), %ecx
movl %eax, (%ecx)
ret
END (__remquof)
weak_alias (__remquof, remquof)

View File

@ -0,0 +1,31 @@
/*
* Written by Ulrich Drepper <drepper@cygnus.com>.
* Based on e_remainder by J.T. Conklin <jtc@netbsd.org>.
* Public domain.
*/
#include <machine/asm.h>
ENTRY(__remquol)
fldt 16(%esp)
fldt 4(%esp)
1: fprem1
fstsw %ax
sahf
jp 1b
fstpl %st(1)
/* Compute the congruent of the quotient. */
movl %eax, %ecx
shrl $8, %eax
shrl $12, %ecx
andl $4, %ecx
andl $3, %eax
orl %eax, %ecx
movl $0xef2960, %eax
shrl %cl, %eax
andl $3, %eax
movl 28(%esp), %ecx
movl %eax, (%ecx)
ret
END (__remquol)
weak_alias (__remquol, remquol)

View File

@ -0,0 +1,47 @@
/* Return quiet nan.
Copyright (C) 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library.
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
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.
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.
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. */
/* XXX The ISO C 9X standard mentions a `n-char-sequence' which is provided
as the argument to this function but I have no clue what this means.
Perhaps it is a description of the bits set in the mantissa. */
#include <math.h>
#ifdef HANDLE_TAGP
# include <stdlib.h>
# include <string.h>
#else
# include <ieee754.h>
#endif
double
nan (const char *tagp)
{
#ifdef HANDLE_TAGP
/* If we ever should have use of the TAGP parameter we will use the
strtod function to analyze it. */
char buf[6 + strlen (tagp)];
sprintf (buf, "NAN(%s)", tagp);
return strtod (buf, NULL);
#else
static const union ieee754_double nan_value =
{ ieee: { mantissa1: 0x1, mantissa0: 0x0, exponent: 0x7ff, negative: 0 } };
return nan_value.d;
#endif
}

View File

@ -0,0 +1,47 @@
/* Return quiet nan.
Copyright (C) 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library.
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
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.
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.
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. */
/* XXX The ISO C 9X standard mentions a `n-char-sequence' which is provided
as the argument to this function but I have no clue what this means.
Perhaps it is a description of the bits set in the mantissa. */
#include <math.h>
#ifdef HANDLE_TAGP
# include <stdlib.h>
# include <string.h>
#else
# include <ieee754.h>
#endif
float
nanf (const char *tagp)
{
#ifdef HANDLE_TAGP
/* If we ever should have use of the TAGP parameter we will use the
strtod function to analyze it. */
char buf[6 + strlen (tagp)];
sprintf (buf, "NAN(%s)", tagp);
return strtof (buf, NULL);
#else
static const union ieee754_float nan_value =
{ ieee: { mantissa: 0x1, exponent: 0xff, negative: 0 } };
return nan_value.f;
#endif
}

View File

@ -0,0 +1,48 @@
/* Return quiet nan.
Copyright (C) 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library.
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
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.
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.
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. */
/* XXX The ISO C 9X standard mentions a `n-char-sequence' which is provided
as the argument to this function but I have no clue what this means.
Perhaps it is a description of the bits set in the mantissa. */
#include <math.h>
#ifdef HANDLE_TAGP
# include <stdlib.h>
# include <string.h>
#else
# include <ieee754.h>
#endif
long double
nanl (const char *tagp)
{
#ifdef HANDLE_TAGP
/* If we ever should have use of the TAGP parameter we will use the
strtod function to analyze it. */
char buf[6 + strlen (tagp)];
sprintf (buf, "NAN(%s)", tagp);
return strtold (buf, NULL);
#else
static const union ieee854_long_double nan_value =
{ ieee: { mantissa1: 0x1, mantissa0: 0x80000000,
exponent: 0x7fff, negative: 0 } };
return nan_value.d;
#endif
}

View File

@ -42,7 +42,7 @@ static char rcsid[] = "$NetBSD: s_nextafter.c,v 1.8 1995/05/10 20:47:58 jtc Exp
if(((ix>=0x7ff00000)&&((ix-0x7ff00000)|lx)!=0) || /* x is nan */ if(((ix>=0x7ff00000)&&((ix-0x7ff00000)|lx)!=0) || /* x is nan */
((iy>=0x7ff00000)&&((iy-0x7ff00000)|ly)!=0)) /* y is nan */ ((iy>=0x7ff00000)&&((iy-0x7ff00000)|ly)!=0)) /* y is nan */
return x+y; return x+y;
if(x==y) return x; /* x=y, return x */ if(x==y) return y; /* x=y, return y */
if((ix|lx)==0) { /* x == 0 */ if((ix|lx)==0) { /* x == 0 */
INSERT_WORDS(x,hy&0x80000000,1); /* return +-minsubnormal */ INSERT_WORDS(x,hy&0x80000000,1); /* return +-minsubnormal */
y = x*x; y = x*x;

View File

@ -37,7 +37,7 @@ static char rcsid[] = "$NetBSD: s_nextafterf.c,v 1.4 1995/05/10 20:48:01 jtc Exp
if((ix>0x7f800000) || /* x is nan */ if((ix>0x7f800000) || /* x is nan */
(iy>0x7f800000)) /* y is nan */ (iy>0x7f800000)) /* y is nan */
return x+y; return x+y;
if(x==y) return x; /* x=y, return x */ if(x==y) return y; /* x=y, return y */
if(ix==0) { /* x == 0 */ if(ix==0) { /* x == 0 */
SET_FLOAT_WORD(x,(hy&0x80000000)|1);/* return +-minsubnormal */ SET_FLOAT_WORD(x,(hy&0x80000000)|1);/* return +-minsubnormal */
y = x*x; y = x*x;

View File

@ -46,7 +46,7 @@ static char rcsid[] = "$NetBSD: $";
if(((ix==0x7fff)&&((hx|lx)|-(hx|lx))!=0) || /* x is nan */ if(((ix==0x7fff)&&((hx|lx)|-(hx|lx))!=0) || /* x is nan */
((iy==0x7fff)&&((hy|ly)|-(hy|ly))!=0)) /* y is nan */ ((iy==0x7fff)&&((hy|ly)|-(hy|ly))!=0)) /* y is nan */
return x+y; return x+y;
if(x==y) return x; /* x=y, return x */ if(x==y) return y; /* x=y, return y */
if((ix|hx|lx)==0) { /* x == 0 */ if((ix|hx|lx)==0) { /* x == 0 */
SET_LDOUBLE_WORDS(x,esx&0x8000,0,1);/* return +-minsubnormal */ SET_LDOUBLE_WORDS(x,esx&0x8000,0,1);/* return +-minsubnormal */
y = x*x; y = x*x;

View File

@ -0,0 +1,106 @@
/* Compute remainder and a congruent to the quotient.
Copyright (C) 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library.
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
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.
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.
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. */
#include <math.h>
#include "math_private.h"
static const double zero = 0.0;
double
__remquo (double x, double y, int *quo)
{
int32_t hx,hp;
u_int32_t sx,lx,lp;
int cquo;
EXTRACT_WORDS (hx, lx, x);
EXTRACT_WORDS (hp, lp, p);
sx = hx & 0x80000000;
qs = (sx ^ (hp & 0x80000000)) >> 31;
hp &= 0x7fffffff;
hx &= 0x7fffffff;
/* Purge off exception values. */
if ((hp | lp) == 0)
return (x * p) / (x * p); /* p = 0 */
if ((hx >= 0x7ff00000) /* x not finite */
|| ((hp >= 0x7ff00000) /* p is NaN */
&& (((hp - 0x7ff00000) | lp) != 0)))
return (x * p) / (x * p);
if (hp <= 0x7fbfffff)
{
x = __ieee754_fmod (x, 8 * p); /* now x < 8p */
if (fabs (x) >= 4 * fabs (p))
cquo += 4;
}
if (((hx - hp) | (lx - lp)) == 0)
{
*quo = qs ? -1 : 1;
return zero * x;
}
x = fabs (x);
p = fabs (p);
cquo = 0;
if (x >= 2 * p)
{
x -= 4 * p;
cquo += 2;
}
if (x >= p)
{
x -= 2 * p;
++cquo;
}
if (hp < 0x00200000)
{
if (x + x > p)
{
x -= p;
if (x + x >= p)
x -= p;
}
}
else
{
double p_half = 0.5 * p;
if(x > p_half)
{
x -= p;
if (x >= p_half)
x -= p;
}
}
*quo = qs ? -cquo : cquo;
GET_HIGH_WORD (hx, x);
SET_HIGH_WORD (x, hx ^ sx);
return x;
}
weak_alias (__remquo, remquo)

View File

@ -0,0 +1,105 @@
/* Compute remainder and a congruent to the quotient.
Copyright (C) 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library.
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
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.
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.
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. */
#include <math.h>
#include "math_private.h"
static const float zero = 0.0;
float
__remquof (float x, float y, int *quo)
{
int32_t hx,hp;
u_int32_t sx;
int cquo;
GET_FLOAT_WORD (hx, x);
GET_FLOAT_WORD (hp, p);
sx = hx & 0x80000000;
qs = (sx ^ (hp & 0x80000000)) >> 31;
hp &= 0x7fffffff;
hx &= 0x7fffffff;
/* Purge off exception values. */
if (hp == 0)
return (x * p) / (x * p); /* p = 0 */
if ((hx >= 0x7f800000) /* x not finite */
|| (hp > 0x7f800000)) /* p is NaN */
return (x * p) / (x * p);
if (hp <= 0x7dffffff)
{
x = __ieee754_fmodf (x, 8 * p); /* now x < 8p */
if (fabs (x) >= 4 * fabs (p))
cquo += 4;
}
if ((hx - hp) == 0)
{
*quo = qs ? -1 : 1;
return zero * x;
}
x = fabsf (x);
p = fabsf (p);
cquo = 0;
if (x >= 2 * p)
{
x -= 4 * p;
cquo += 2;
}
if (x >= p)
{
x -= 2 * p;
++cquo;
}
if (hp < 0x01000000)
{
if (x + x > p)
{
x -= p;
if (x + x >= p)
x -= p;
}
}
else
{
float p_half = 0.5 * p;
if(x > p_half)
{
x -= p;
if (x >= p_half)
x -= p;
}
}
*quo = qs ? -cquo : cquo;
GET_FLOAT_WORD (hx, x);
SET_FLOAT_WORD (x, hx ^ sx);
return x;
}
weak_alias (__remquof, remquof)

View File

@ -0,0 +1,106 @@
/* Compute remainder and a congruent to the quotient.
Copyright (C) 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library.
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
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.
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.
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. */
#include <math.h>
#include "math_private.h"
static const double zero = 0.0;
long double
__remquol (long double x, long double y, int *quo)
{
int32_t ex,ep,hx,hp;
u_int32_t sx,lx,lp;
int cquo;
GET_LDOUBLE_WORDS (ex, hx, lx, x);
GET_LDOUBLE_WORDS (ep, hp, lp, p);
sx = ex & 0x8000;
qs = (sx ^ (ep & 0x8000)) >> 15;
ep &= 0x7fff;
ex &= 0x7fff;
/* Purge off exception values. */
if ((ep | hp | lp) == 0)
return (x * p) / (x * p); /* p = 0 */
if ((ex == 0x7fff) /* x not finite */
|| ((ep == 0x7fff) /* p is NaN */
&& ((hp | lp) != 0)))
return (x * p) / (x * p);
if (ep <= 0x7ffb)
{
x = __ieee754_fmodl (x, 8 * p); /* now x < 8p */
if (fabsl (x) >= 4 * fabsl (p))
cquo += 4;
}
if (((ex - ep) | (hx - hp) | (lx - lp)) == 0)
{
*quo = qs ? -1 : 1;
return zero * x;
}
x = fabsl (x);
p = fabsl (p);
cquo = 0;
if (x >= 2 * p)
{
x -= 4 * p;
cquo += 2;
}
if (x >= p)
{
x -= 2 * p;
++cquo;
}
if (ep < 0x0002)
{
if (x + x > p)
{
x -= p;
if (x + x >= p)
x -= p;
}
}
else
{
long double p_half = 0.5 * p;
if(x > p_half)
{
x -= p;
if (x >= p_half)
x -= p;
}
}
*quo = qs ? -cquo : cquo;
GET_LDOUBLE_EXP (ex, x);
SET_LDOUBLE_EXP (x, ex ^ sx);
return x;
}
weak_alias (__remquol, remquol)

View File

@ -54,7 +54,7 @@ TWO64[2]={
GET_LDOUBLE_WORDS(se,i0,i1,x); GET_LDOUBLE_WORDS(se,i0,i1,x);
sx = (se>>15)&1; sx = (se>>15)&1;
j0 = (se&0x7fff)-0x3fff; j0 = (se&0x7fff)-0x3fff;
if(j0<32) { if(j0<31) {
if(j0<0) { if(j0<0) {
if(((se&0x7fff)|i0|i1)==0) return x; if(((se&0x7fff)|i0|i1)==0) return x;
i1 |= i0; i1 |= i0;
@ -67,7 +67,7 @@ TWO64[2]={
SET_LDOUBLE_EXP(t,(i0&0x7fff)|(sx<<15)); SET_LDOUBLE_EXP(t,(i0&0x7fff)|(sx<<15));
return t; return t;
} else { } else {
i = (0xffffffff)>>j0; i = (0x7fffffff)>>j0;
if(((i0&i)|i1)==0) return x; /* x is integral */ if(((i0&i)|i1)==0) return x; /* x is integral */
i>>=1; i>>=1;
if(((i0&i)|i1)!=0) { if(((i0&i)|i1)!=0) {

View File

@ -0,0 +1,61 @@
/* Truncate argument to nearest integral value not larger than the argument.
Copyright (C) 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library.
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
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.
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.
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. */
#include <math.h>
#include "math_private.h"
double
__trunc (double x)
{
int32_t i0, j0;
u_int32_t i1;
int sx;
EXTRACT_WORDS (i0, i1, x);
sx = i0 & 0x80000000;
j0 = ((i0 >> 20) & 0x7ff) - 0x3ff;
if (j0 < 20)
{
if (j0 < 0)
/* The magnitude of the number is < 1 so the result is +-0. */
INSERT_WORDS (x, sx, 0);
else
INSERT_WORDS (x, sx | i0 & ~(0x000fffff >> j0), 0);
}
else if (j0 > 51)
{
if (j0 == 0x400)
/* x is inf or NaN. */
return x + x;
}
else
{
INSERT_WORDS (x, i0, i1 & ~(0xffffffffu >> (j0 - 20)));
}
return x;
}
weak_alias (__trunc, trunc)
#ifdef NO_LONG_DOUBLE
strong_alias (__trunc, __truncl)
weak_alias (__trunc, truncl)
#endif

View File

@ -0,0 +1,52 @@
/* Truncate argument to nearest integral value not larger than the argument.
Copyright (C) 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library.
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
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.
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.
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. */
#include <math.h>
#include "math_private.h"
float
__truncf (float x)
{
int32_t i0, j0;
int sx;
GET_FLOAT_WORD (i0, x);
sx = i0 & 0x80000000;
j0 = ((i0 >> 23) & 0xff) - 0x7f;
if (j0 < 23)
{
if (j0 < 0)
/* The magnitude of the number is < 1 so the result is +-0. */
SET_FLOAT_WORD (x, sx);
else
SET_FLOAT_WORD (x, sx | i0 & ~(0x007fffff >> j0));
}
else
{
if (j0 == 0x80)
/* x is inf or NaN. */
return x + x;
}
return x;
}
weak_alias (__truncf, truncf)

View File

@ -0,0 +1,57 @@
/* Truncate argument to nearest integral value not larger than the argument.
Copyright (C) 1997 Free Software Foundation, Inc.
This file is part of the GNU C Library.
Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
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.
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.
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. */
#include <math.h>
#include "math_private.h"
long double
__truncl (long double x)
{
int32_t i0, j0;
u_int32_t se, i1;
int sx;
GET_LDOUBLE_WORDS (se, i0, i1, x);
sx = se & 0x8000;
j0 = se & 0x7fff;
if (j0 < 31)
{
if (j0 < 0)
/* The magnitude of the number is < 1 so the result is +-0. */
SET_LDOUBLE_WORDS (x, sx, 0, 0);
else
SET_LDOUBLE_WORDS (x, se, i0 & ~(0x7fffffff >> j0), 0);
}
else if (j0 > 63)
{
if (j0 == 0x4000)
/* x is inf or NaN. */
return x + x;
}
else
{
SET_LDOUBLE_WORDS (x, se, i0, i1 & ~(0xffffffffu >> (j0 - 31)));
}
return x;
}
weak_alias (__truncl, truncl)

View File

@ -104,93 +104,66 @@ elf_machine_rela (struct link_map *map, const Elf32_Rela *reloc,
Elf32_Addr *const reloc_addr = (void *) (map->l_addr + reloc->r_offset); Elf32_Addr *const reloc_addr = (void *) (map->l_addr + reloc->r_offset);
Elf32_Addr loadbase; Elf32_Addr loadbase;
switch (ELF32_R_TYPE (reloc->r_info)) if (ELF32_R_TYPE (reloc->r_info) == R_SPARC_RELATIVE)
*reloc_addr += map->l_addr + reloc->r_addend;
else
{ {
case R_SPARC_COPY: Elf32_Addr value;
loadbase = RESOLVE (&sym, version, DL_LOOKUP_NOEXEC); if (sym->st_shndx != SHN_UNDEF &&
memcpy (reloc_addr, (void *) (loadbase + sym->st_value), sym->st_size); ELF32_ST_BIND (sym->st_info) == STB_LOCAL)
break; value = map->l_addr;
case R_SPARC_GLOB_DAT: else
case R_SPARC_32:
loadbase = RESOLVE (&sym, version, 0);
*reloc_addr = ((sym ? (loadbase + sym->st_value) : 0)
+ reloc->r_addend);
break;
case R_SPARC_JMP_SLOT:
loadbase = RESOLVE (&sym, version, DL_LOOKUP_NOPLT);
{
Elf32_Addr value = ((sym ? (loadbase + sym->st_value) : 0)
+ reloc->r_addend);
reloc_addr[1] = OPCODE_SETHI_G1 | (value >> 10);
reloc_addr[2] = OPCODE_JMP_G1 | (value & 0x3ff);
}
break;
case R_SPARC_8:
loadbase = RESOLVE (&sym, version, 0);
*(char *) reloc_addr = ((sym ? (loadbase + sym->st_value) : 0)
+ reloc->r_addend);
break;
case R_SPARC_16:
loadbase = RESOLVE (&sym, version, 0);
*(short *) reloc_addr = ((sym ? (loadbase + sym->st_value) : 0)
+ reloc->r_addend);
break;
case R_SPARC_RELATIVE:
*reloc_addr += map->l_addr + reloc->r_addend;
break;
case R_SPARC_DISP8:
loadbase = RESOLVE (&sym, version, 0);
*(char *) reloc_addr = ((sym ? (loadbase + sym->st_value) : 0)
+ reloc->r_addend
- (Elf32_Addr) reloc_addr);
break;
case R_SPARC_DISP16:
loadbase = RESOLVE (&sym, version, 0);
*(short *) reloc_addr = ((sym ? (loadbase + sym->st_value) : 0)
+ reloc->r_addend
- (Elf32_Addr) reloc_addr);
break;
case R_SPARC_DISP32:
loadbase = RESOLVE (&sym, version, 0);
*reloc_addr = ((sym ? (loadbase + sym->st_value) : 0)
+ reloc->r_addend
- (Elf32_Addr) reloc_addr);
break;
case R_SPARC_LO10:
{ {
unsigned int saddr; value = RESOLVE (&sym, version, ELF32_R_TYPE (reloc->r_info));
if (sym)
loadbase = RESOLVE (&sym, version, 0); value += sym->st_value;
saddr = (loadbase ? loadbase : map->l_addr) + reloc->r_addend;
*reloc_addr = (*reloc_addr & ~0x3ff) | (saddr & 0x3ff);
} }
break; value += reloc->r_addend; /* Assume copy relocs have zero addend. */
case R_SPARC_WDISP30:
{
unsigned int saddr;
loadbase = RESOLVE (&sym, version, 0); switch (ELF32_R_TYPE (reloc->r_info))
saddr = (loadbase ? loadbase : map->l_addr) + reloc->r_addend; {
*reloc_addr = ((*reloc_addr & 0xc0000000) case R_SPARC_COPY:
| ((saddr - (unsigned int) reloc_addr)>>2)); memcpy (reloc_addr, (void *) value, sym->st_size);
} break;
break; case R_SPARC_GLOB_DAT:
case R_SPARC_HI22: case R_SPARC_32:
{ *reloc_addr = value;
unsigned int saddr; break;
case R_SPARC_JMP_SLOT:
loadbase = RESOLVE (&sym, version, 0); reloc_addr[1] = OPCODE_SETHI_G1 | (value >> 10);
saddr = (loadbase ? loadbase : map->l_addr) + reloc->r_addend; reloc_addr[2] = OPCODE_JMP_G1 | (value & 0x3ff);
break;
*reloc_addr = (*reloc_addr & 0xffc00000)|(saddr >> 10); case R_SPARC_8:
} *(char *) reloc_addr = value;
break; break;
case R_SPARC_NONE: /* Alright, Wilbur. */ case R_SPARC_16:
break; *(short *) reloc_addr = value;
default: break;
assert (! "unexpected dynamic reloc type"); case R_SPARC_DISP8:
break; *(char *) reloc_addr = (value - (Elf32_Addr) reloc_addr);
break;
case R_SPARC_DISP16:
*(short *) reloc_addr = (value - (Elf32_Addr) reloc_addr);
break;
case R_SPARC_DISP32:
*reloc_addr = (value - (Elf32_Addr) reloc_addr);
break;
case R_SPARC_LO10:
*reloc_addr = (*reloc_addr & ~0x3ff) | (value & 0x3ff);
break;
case R_SPARC_WDISP30:
*reloc_addr = ((*reloc_addr & 0xc0000000)
| ((value - (unsigned int) reloc_addr) >> 2));
break;
case R_SPARC_HI22:
*reloc_addr = (*reloc_addr & 0xffc00000) | (value >> 10);
break;
case R_SPARC_NONE: /* Alright, Wilbur. */
break;
default:
assert (! "unexpected dynamic reloc type");
break;
}
} }
} }
@ -213,7 +186,14 @@ elf_machine_lazy_rel (struct link_map *map, const Elf32_Rela *reloc)
/* Nonzero iff TYPE describes relocation of a PLT entry, so /* Nonzero iff TYPE describes relocation of a PLT entry, so
PLT entries should not be allowed to define the value. */ PLT entries should not be allowed to define the value. */
#define elf_machine_pltrel_p(type) ((type) == R_SPARC_JMP_SLOT) #define elf_machine_lookup_noexec_p(type) ((type) == R_SPARC_COPY)
/* Nonzero iff TYPE describes relocation of a PLT entry, so
PLT entries should not be allowed to define the value. */
#define elf_machine_lookup_noplt_p(type) ((type) == R_SPARC_JMP_SLOT)
/* A reloc type used for ld.so cmdline arg lookups to reject PLT entries. */
#define ELF_MACHINE_RELOC_NOPLT R_SPARC_JMP_SLOT
/* The SPARC never uses Elf32_Rel relocations. */ /* The SPARC never uses Elf32_Rel relocations. */
#define ELF_MACHINE_NO_REL 1 #define ELF_MACHINE_NO_REL 1

View File

@ -112,6 +112,10 @@ struct termios
#define VT1 00200000 #define VT1 00200000
#define XTABS 01000000 /* Hmm.. Linux/i386 considers this part of TABDLY.. */ #define XTABS 01000000 /* Hmm.. Linux/i386 considers this part of TABDLY.. */
/* On Linux there is no OXTABS bit defined. Take it as an alias for
XTABS. */
#define OXTABS XTABS
/* c_cflag bit meaning */ /* c_cflag bit meaning */
#define CBAUD 0000037 #define CBAUD 0000037
#define B0 0000000 /* hang up */ #define B0 0000000 /* hang up */

View File

@ -105,6 +105,10 @@ struct termios
#define FF0 0000000 #define FF0 0000000
#define FF1 0100000 #define FF1 0100000
/* On Linux there is no OXTABS bit defined. Take it as an alias for
XTABS. */
#define OXTABS XTABS
/* c_cflag bit meaning */ /* c_cflag bit meaning */
#define CBAUD 0010017 #define CBAUD 0010017
#define B0 0000000 /* hang up */ #define B0 0000000 /* hang up */

View File

@ -48,11 +48,7 @@
* Defaults on "first" open. * Defaults on "first" open.
*/ */
#define TTYDEF_IFLAG (BRKINT | ISTRIP | ICRNL | IMAXBEL | IXON | IXANY) #define TTYDEF_IFLAG (BRKINT | ISTRIP | ICRNL | IMAXBEL | IXON | IXANY)
#ifdef OXTABS
#define TTYDEF_OFLAG (OPOST | ONLCR | OXTABS) #define TTYDEF_OFLAG (OPOST | ONLCR | OXTABS)
#else
#define TTYDEF_OFLAG (OPOST | ONLCR | XTABS)
#endif
#define TTYDEF_LFLAG (ECHO | ICANON | ISIG | IEXTEN | ECHOE|ECHOKE|ECHOCTL) #define TTYDEF_LFLAG (ECHO | ICANON | ISIG | IEXTEN | ECHOE|ECHOKE|ECHOCTL)
#define TTYDEF_CFLAG (CREAD | CS7 | PARENB | HUPCL) #define TTYDEF_CFLAG (CREAD | CS7 | PARENB | HUPCL)
#define TTYDEF_SPEED (B9600) #define TTYDEF_SPEED (B9600)

View File

@ -81,11 +81,14 @@ enum __itimer_which
{ {
/* Timers run in real time. */ /* Timers run in real time. */
ITIMER_REAL = 0, ITIMER_REAL = 0,
#define ITIMER_REAL ITIMER_REAL
/* Timers run only when the process is executing. */ /* Timers run only when the process is executing. */
ITIMER_VIRTUAL = 1, ITIMER_VIRTUAL = 1,
#define ITIMER_VIRTUAL ITIMER_VIRTUAL
/* Timers run when the process is executing and when /* Timers run when the process is executing and when
the system is executing on behalf of the process. */ the system is executing on behalf of the process. */
ITIMER_PROF = 2 ITIMER_PROF = 2
#define ITIMER_PROF ITIMER_PROF
}; };
/* Type of the second argument to `getitimer' and /* Type of the second argument to `getitimer' and