diff options
| author | Waldemar Brodkorb <wbx@openadk.org> | 2017-05-24 20:49:02 +0200 | 
|---|---|---|
| committer | Waldemar Brodkorb <wbx@openadk.org> | 2017-06-23 23:46:04 +0200 | 
| commit | 041cdc2769407c4d3869b218ad7ee7638e1c306e (patch) | |
| tree | 1ea25250d74dcb230cf5feee99226fc080dbd678 /libc | |
| parent | 58a5ba12bffad5916d9897c2870fc483f1db8282 (diff) | |
sparc64: add basic support
No NPTL, no LDSO support.
Bootup with Busybox Ash in Qemu working.
Testuite shows only two failures, but mksh continue/break
support doesn't work.
Diffstat (limited to 'libc')
111 files changed, 13472 insertions, 1 deletions
| diff --git a/libc/sysdeps/linux/common/bits/kernel_sigaction.h b/libc/sysdeps/linux/common/bits/kernel_sigaction.h index 5c8726058..21e4c9296 100644 --- a/libc/sysdeps/linux/common/bits/kernel_sigaction.h +++ b/libc/sysdeps/linux/common/bits/kernel_sigaction.h @@ -19,6 +19,14 @@ struct old_kernel_sigaction {  	unsigned long sa_flags;  	void (*sa_restorer)(void);  }; + +/* This is the sigaction structure from the Linux 2.1.68 kernel.  */ +struct kernel_sigaction { +	__sighandler_t k_sa_handler; +	unsigned long sa_flags; +	void (*sa_restorer) (void); +	sigset_t sa_mask; +};  #endif  #endif /* _BITS_SIGACTION_STRUCT_H */ diff --git a/libc/sysdeps/linux/common/mmap.c b/libc/sysdeps/linux/common/mmap.c index cafe756bf..b74fc19ef 100644 --- a/libc/sysdeps/linux/common/mmap.c +++ b/libc/sysdeps/linux/common/mmap.c @@ -21,6 +21,7 @@ static _syscall6(void *, _mmap, void *, addr, size_t, len,  #elif defined __NR_mmap2 && defined _syscall6 +  # include <errno.h>  # include <bits/uClibc_page.h>  # ifndef MMAP2_PAGE_SHIFT @@ -51,7 +52,6 @@ static void *_mmap(void *addr, size_t len, int prot, int flags,  }  #elif defined __NR_mmap -  # define __NR___syscall_mmap __NR_mmap  static __inline__ _syscall1(void *, __syscall_mmap, unsigned long *, buffer) diff --git a/libc/sysdeps/linux/sparc64/Makefile b/libc/sysdeps/linux/sparc64/Makefile new file mode 100644 index 000000000..633c91f3e --- /dev/null +++ b/libc/sysdeps/linux/sparc64/Makefile @@ -0,0 +1,13 @@ +# Makefile for uClibc +# +# Copyright (C) 2000-2005 Erik Andersen <andersen@uclibc.org> +# +# Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball. +# + +top_srcdir=../../../../ +top_builddir=../../../../ +all: objs +include $(top_builddir)Rules.mak +include Makefile.arch +include $(top_srcdir)Makerules diff --git a/libc/sysdeps/linux/sparc64/Makefile.arch b/libc/sysdeps/linux/sparc64/Makefile.arch new file mode 100644 index 000000000..37b539b3b --- /dev/null +++ b/libc/sysdeps/linux/sparc64/Makefile.arch @@ -0,0 +1,18 @@ +# Makefile for uClibc-ng +# +# Copyright (C) 2017 Waldemar Brodkorb <wbx@uclibc-ng.org> +# Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball. + +CSRC-y := __syscall_error.c sigaction.c +SSRC-y := __longjmp.S setjmp.S bsd-setjmp.S bsd-_setjmp.S brk.S fork.S \ +	syscall.S pipe.S vfork.S clone.S + +CSRC-y += $(addprefix soft-fp/, \ +	qp_add.c qp_cmp.c qp_cmpe.c qp_div.c qp_dtoq.c qp_feq.c qp_fge.c \ +	qp_fgt.c qp_fle.c qp_flt.c qp_fne.c qp_itoq.c qp_mul.c qp_neg.c \ +	qp_qtod.c qp_qtoi.c qp_qtos.c qp_qtoui.c qp_qtoux.c qp_qtox.c \ +	qp_sqrt.c qp_stoq.c qp_sub.c qp_uitoq.c qp_uxtoq.c qp_xtoq.c qp_util.c) + +CSRC-$(UCLIBC_HAS_CONTEXT_FUNCS) += makecontext.c +SSRC-$(UCLIBC_HAS_CONTEXT_FUNCS) += getcontext.S setcontext.S swapcontext.S \ +				    __start_context.S diff --git a/libc/sysdeps/linux/sparc64/__longjmp.S b/libc/sysdeps/linux/sparc64/__longjmp.S new file mode 100644 index 000000000..79cefcc9d --- /dev/null +++ b/libc/sysdeps/linux/sparc64/__longjmp.S @@ -0,0 +1,40 @@ +/* Copyright (C) 1997-2017 Free Software Foundation, Inc. +   Contributed by Richard Henderson (rth@tamu.edu). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <sysdep.h> + +/* Offsets into the jmp_buf structure.  */ + +#define O_mask_was_saved	512 +#define O_gregs			32 +#define O_g1			(O_gregs + 4*8) + +ENTRY(__longjmp) + +	/* Modify the context with the value we want to return.  */ +	movre	%o1, 1, %o1 +	stx	%o1, [%o0 + O_g1] + +	/* Let setcontext know if we want to modify the current sigmask. */ +	ld	[%o0 + O_mask_was_saved], %o1 + +	/* And bamf back to where we belong!  */ +	ta	0x6f + +END(__longjmp) + +libc_hidden_def (__longjmp) diff --git a/libc/sysdeps/linux/sparc64/__start_context.S b/libc/sysdeps/linux/sparc64/__start_context.S new file mode 100644 index 000000000..7054845da --- /dev/null +++ b/libc/sysdeps/linux/sparc64/__start_context.S @@ -0,0 +1,35 @@ +/* Copyright (C) 2012-2017 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <sysdep.h> + +	.text + +/* This is the helper code which gets called if a function which is +   registered with 'makecontext' returns.  In this case we have to +   install the context listed in the uc_link element of the context +   'makecontext' manipulated at the time of the 'makecontext' call. +   If the pointer is NULL the process must terminate.  */ + +ENTRY(__start_context) +	brz,pn	%i0, 1f +	 mov	1, %o1 +	call	__setcontext +	 mov	%i0, %o0 +1:	call	HIDDEN_JUMPTARGET(exit) +	 mov	0, %o0 +	unimp	0 +END(__start_context) diff --git a/libc/sysdeps/linux/sparc64/__syscall_error.c b/libc/sysdeps/linux/sparc64/__syscall_error.c new file mode 100644 index 000000000..5e109a83b --- /dev/null +++ b/libc/sysdeps/linux/sparc64/__syscall_error.c @@ -0,0 +1,18 @@ +/* Wrapper for setting errno. + * + * Copyright (C) 2000-2006 Erik Andersen <andersen@uclibc.org> + * + * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball. + */ + +#include <errno.h> +#include <features.h> + +/* This routine is jumped to by all the syscall handlers, to stash + * an error number into errno.  */ +int __syscall_error(int err_no) attribute_hidden; +int __syscall_error(int err_no) +{ +	__set_errno(err_no); +	return -1; +} diff --git a/libc/sysdeps/linux/sparc64/bits/endian.h b/libc/sysdeps/linux/sparc64/bits/endian.h new file mode 100644 index 000000000..8acfdf5df --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/endian.h @@ -0,0 +1,12 @@ +/* Sparc is big-endian, but v9 supports endian conversion on loads/stores +   and GCC supports such a mode.  Be prepared.  */ + +#ifndef _ENDIAN_H +# error "Never use <bits/endian.h> directly; include <endian.h> instead." +#endif + +#ifdef __LITTLE_ENDIAN__ +# define __BYTE_ORDER __LITTLE_ENDIAN +#else +# define __BYTE_ORDER __BIG_ENDIAN +#endif diff --git a/libc/sysdeps/linux/sparc64/bits/epoll.h b/libc/sysdeps/linux/sparc64/bits/epoll.h new file mode 100644 index 000000000..3939f1936 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/epoll.h @@ -0,0 +1,28 @@ +/* Copyright (C) 2002-2012 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef	_SYS_EPOLL_H +# error "Never use <bits/epoll.h> directly; include <sys/epoll.h> instead." +#endif + +/* Flags to be passed to epoll_create1.  */ +enum +  { +    EPOLL_CLOEXEC = 0x400000, +#define EPOLL_CLOEXEC EPOLL_CLOEXEC +    EPOLL_NONBLOCK = 0x004000 +#define EPOLL_NONBLOCK EPOLL_NONBLOCK +  }; diff --git a/libc/sysdeps/linux/sparc64/bits/eventfd.h b/libc/sysdeps/linux/sparc64/bits/eventfd.h new file mode 100644 index 000000000..e2eb4d27d --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/eventfd.h @@ -0,0 +1,30 @@ +/* Copyright (C) 2007-2013 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef	_SYS_EVENTFD_H +# error "Never use <bits/eventfd.h> directly; include <sys/eventfd.h> instead." +#endif + +/* Flags for eventfd.  */ +enum +  { +    EFD_SEMAPHORE = 0x000001, +#define EFD_SEMAPHORE EFD_SEMAPHORE +    EFD_CLOEXEC = 0x400000, +#define EFD_CLOEXEC EFD_CLOEXEC +    EFD_NONBLOCK = 0x004000 +#define EFD_NONBLOCK EFD_NONBLOCK +  }; diff --git a/libc/sysdeps/linux/sparc64/bits/fcntl.h b/libc/sysdeps/linux/sparc64/bits/fcntl.h new file mode 100644 index 000000000..dad3470b8 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/fcntl.h @@ -0,0 +1,247 @@ +/* O_*, F_*, FD_* bit values for Linux/SPARC64. +   Copyright (C) 1995-2017 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _FCNTL_H +# error "Never use <bits/fcntl.h> directly; include <fcntl.h> instead." +#endif + +#include <sys/types.h> +#ifdef __USE_GNU +# include <bits/uio.h> +#endif + +/* open/fcntl - O_SYNC is only implemented on blocks devices and on files +   located on an ext2 file system */ +#define O_RDONLY	0x0000 +#define O_WRONLY	0x0001 +#define O_RDWR		0x0002 +#define O_ACCMODE	0x0003 +#define O_APPEND	0x0008 +#define O_ASYNC		0x0040 +#define O_CREAT		0x0200	/* not fcntl */ +#define O_TRUNC		0x0400	/* not fcntl */ +#define O_EXCL		0x0800	/* not fcntl */ +#define O_SYNC		0x2000 +#define O_NONBLOCK	0x4000 +#define O_NDELAY	(0x0004 | O_NONBLOCK) +#define O_NOCTTY	0x8000	/* not fcntl */ + +#ifdef __USE_GNU +# define O_DIRECTORY	0x10000 /* must be a directory */ +# define O_NOFOLLOW	0x20000 /* don't follow links */ +# define O_DIRECT      0x100000 /* direct disk access hint */ +# define O_NOATIME     0x200000 /* Do not set atime.  */ +# define O_CLOEXEC     0x400000 /* Set close_on_exit.  */ +# define O_PATH       0x1000000 /* Resolve pathname but do not open file.  */ +#endif + +#ifdef __USE_LARGEFILE64 +# define O_LARGEFILE	0 +#endif + +/* For now Linux has no synchronisity options for data and read +   operations.  We define the symbols here but let them do the same as +   O_SYNC since this is a superset.  */ +#if defined __USE_POSIX199309 || defined __USE_UNIX98 +# define O_DSYNC	O_SYNC	/* Synchronize data.  */ +# define O_RSYNC	O_SYNC	/* Synchronize read operations.  */ +#endif + +/* For now Linux has synchronisity options for data and read operations. +   We define the symbols here but let them do the same as O_SYNC since +   this is a superset.  */ +#if defined __USE_POSIX199309 || defined __USE_UNIX98 +# define O_DSYNC        O_SYNC  /* Synchronize data.  */ +# define O_RSYNC        O_SYNC  /* Synchronize read operations.  */ +#endif + +/* Values for the second argument to `fcntl'.  */ +#define F_DUPFD		0	/* Duplicate file descriptor.  */ +#define F_GETFD		1	/* Get file descriptor flags.  */ +#define F_SETFD		2	/* Set file descriptor flags.  */ +#define F_GETFL		3	/* Get file status flags.  */ +#define F_SETFL		4	/* Set file status flags.  */ +#if defined __USE_BSD || defined __USE_UNIX98 +# define F_GETOWN	5	/* Get owner of socket (receiver of SIGIO).  */ +# define F_SETOWN	6	/* Set owner of socket (receiver of SIGIO).  */ +#endif +#ifndef __USE_FILE_OFFSET64 +# define F_GETLK	7	/* Get record locking info.  */ +# define F_SETLK	8	/* Set record locking info (non-blocking).  */ +# define F_SETLKW	9	/* Set record locking info (blocking).  */ +#else +# define F_GETLK	F_GETLK64  /* Get record locking info.  */ +# define F_SETLK	F_SETLK64  /* Set record locking info (non-blocking).*/ +# define F_SETLKW	F_SETLKW64 /* Set record locking info (blocking).  */ +#endif + +#ifdef __USE_GNU +# define F_SETSIG	10	/* Set number of signal to be sent.  */ +# define F_GETSIG	11	/* Get number of signal to be sent.  */ +#endif + +#ifdef __USE_GNU +# define F_SETLEASE     1024	/* Set a lease.  */ +# define F_GETLEASE     1025	/* Enquire what lease is active.  */ +# define F_NOTIFY       1026	/* Request notfications on a directory.  */ +# define F_DUPFD_CLOEXEC 1030	/* Duplicate file descriptor with +				   close-on-exit set on new fd.  */ +# define F_SETPIPE_SZ	1031    /* Set pipe page size array.  */ +# define F_GETPIPE_SZ	1032    /* Get pipe page size array.  */ +#endif + +#define F_GETLK64	7	/* Get record locking info.  */ +#define F_SETLK64	8	/* Set record locking info (non-blocking).  */ +#define F_SETLKW64	9	/* Set record locking info (blocking).  */ + +/* for F_[GET|SET]FD */ +#define FD_CLOEXEC	1	/* actually anything with low bit set goes */ + +/* For posix fcntl() and `l_type' field of a `struct flock' for lockf().  */ +#define F_RDLCK		1	/* Read lock.  */ +#define F_WRLCK		2	/* Write lock.  */ +#define F_UNLCK		3	/* Remove lock.  */ + +/* for old implementation of bsd flock () */ +#define F_EXLCK		4	/* or 3 */ +#define F_SHLCK		8	/* or 4 */ + +#ifdef __USE_BSD +/* Operations for bsd flock(), also used by the kernel implementation */ +# define LOCK_SH	1	/* shared lock */ +# define LOCK_EX	2	/* exclusive lock */ +# define LOCK_NB	4	/* or'd with one of the above to prevent +				   blocking */ +# define LOCK_UN	8	/* remove lock */ +#endif + +#ifdef __USE_GNU +# define LOCK_MAND	32	/* This is a mandatory flock:	*/ +# define LOCK_READ	64	/* ... which allows concurrent read operations.	 */ +# define LOCK_WRITE	128	/* ... which allows concurrent write operations.  */ +# define LOCK_RW	192	/* ... Which allows concurrent read & write operations.	 */ +#endif + +#ifdef __USE_GNU +/* Types of directory notifications that may be requested with F_NOTIFY.  */ +# define DN_ACCESS	0x00000001	/* File accessed.  */ +# define DN_MODIFY	0x00000002	/* File modified.  */ +# define DN_CREATE	0x00000004	/* File created.  */ +# define DN_DELETE	0x00000008	/* File removed.  */ +# define DN_RENAME	0x00000010	/* File renamed.  */ +# define DN_ATTRIB	0x00000020	/* File changed attibutes.  */ +# define DN_MULTISHOT	0x80000000	/* Don't remove notifier.  */ +#endif + +struct flock +  { +    short int l_type;	/* Type of lock: F_RDLCK, F_WRLCK, or F_UNLCK.  */ +    short int l_whence;	/* Where `l_start' is relative to (like `lseek').  */ +#ifndef __USE_FILE_OFFSET64 +    __off_t l_start;	/* Offset where the lock begins.  */ +    __off_t l_len;	/* Size of the locked area; zero means until EOF.  */ +#else +    __off64_t l_start;	/* Offset where the lock begins.  */ +    __off64_t l_len;	/* Size of the locked area; zero means until EOF.  */ +#endif +    __pid_t l_pid;	/* Process holding the lock.  */ +    short int __unused; +  }; + +#ifdef __USE_LARGEFILE64 +struct flock64 +  { +    short int l_type;	/* Type of lock: F_RDLCK, F_WRLCK, or F_UNLCK.  */ +    short int l_whence;	/* Where `l_start' is relative to (like `lseek').  */ +    __off64_t l_start;	/* Offset where the lock begins.  */ +    __off64_t l_len;	/* Size of the locked area; zero means until EOF.  */ +    __pid_t l_pid;	/* Process holding the lock.  */ +    short int __unused; +  }; +#endif + +/* Define some more compatibility macros to be backward compatible with +   BSD systems which did not managed to hide these kernel macros.  */ +#ifdef	__USE_BSD +# define FAPPEND	O_APPEND +# define FFSYNC		O_FSYNC +# define FASYNC		O_ASYNC +# define FNONBLOCK	O_NONBLOCK +# define FNDELAY	O_NDELAY +#endif /* Use BSD.  */ + +/* Advise to `posix_fadvise'.  */ +#ifdef __USE_XOPEN2K +# define POSIX_FADV_NORMAL	0 /* No further special treatment.  */ +# define POSIX_FADV_RANDOM	1 /* Expect random page references.  */ +# define POSIX_FADV_SEQUENTIAL	2 /* Expect sequential page references.  */ +# define POSIX_FADV_WILLNEED	3 /* Will need these pages.  */ +# define POSIX_FADV_DONTNEED	4 /* Don't need these pages.  */ +# define POSIX_FADV_NOREUSE	5 /* Data will be accessed once.  */ +#endif + + +#if defined __USE_GNU && defined __UCLIBC_LINUX_SPECIFIC__ +/* Flags for SYNC_FILE_RANGE.  */ +# define SYNC_FILE_RANGE_WAIT_BEFORE	1 /* Wait upon writeout of all pages +					     in the range before performing the +					     write.  */ +# define SYNC_FILE_RANGE_WRITE		2 /* Initiate writeout of all those +					     dirty pages in the range which are +					     not presently under writeback.  */ +# define SYNC_FILE_RANGE_WAIT_AFTER	4 /* Wait upon writeout of all pages in +					     the range after performing the +					     write.  */ + +/* Flags for SPLICE and VMSPLICE.  */ +# define SPLICE_F_MOVE		1	/* Move pages instead of copying.  */ +# define SPLICE_F_NONBLOCK	2	/* Don't block on the pipe splicing +					   (but we may still block on the fd +					   we splice from/to).  */ +# define SPLICE_F_MORE		4	/* Expect more data.  */ +# define SPLICE_F_GIFT		8	/* Pages passed in are a gift.  */ +#endif + +__BEGIN_DECLS + +#if defined __USE_GNU && defined __UCLIBC_LINUX_SPECIFIC__ + +/* Provide kernel hint to read ahead.  */ +extern ssize_t readahead (int __fd, __off64_t __offset, size_t __count) +    __THROW; + + +/* Selective file content synch'ing.  */ +extern int sync_file_range (int __fd, __off64_t __from, __off64_t __to, +			    unsigned int __flags); + +/* Splice address range into a pipe.  */ +extern ssize_t vmsplice (int __fdout, const struct iovec *__iov, +			 size_t __count, unsigned int __flags); + +/* Splice two files together.  */ +extern ssize_t splice (int __fdin, __off64_t *__offin, int __fdout, +		       __off64_t *__offout, size_t __len, +		       unsigned int __flags); + +/* In-kernel implementation of tee for pipe buffers.  */ +extern ssize_t tee (int __fdin, int __fdout, size_t __len, +		    unsigned int __flags); + +#endif +__END_DECLS + diff --git a/libc/sysdeps/linux/sparc64/bits/inotify.h b/libc/sysdeps/linux/sparc64/bits/inotify.h new file mode 100644 index 000000000..41ca6278a --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/inotify.h @@ -0,0 +1,28 @@ +/* Copyright (C) 2005-2012 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef	_SYS_INOTIFY_H +# error "Never use <bits/inotify.h> directly; include <sys/inotify.h> instead." +#endif + +/* Flags for the parameter of inotify_init1.  */ +enum +  { +    IN_CLOEXEC = 0x400000, +#define IN_CLOEXEC IN_CLOEXEC +    IN_NONBLOCK = 0x004000 +#define IN_NONBLOCK IN_NONBLOCK +  }; diff --git a/libc/sysdeps/linux/sparc64/bits/ioctls.h b/libc/sysdeps/linux/sparc64/bits/ioctls.h new file mode 100644 index 000000000..49573ed4f --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/ioctls.h @@ -0,0 +1,35 @@ +/* Copyright (C) 1996, 1997 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_IOCTL_H +# error "Never use <bits/ioctls.h> directly; include <sys/ioctl.h> instead." +#endif + +/* Use the definitions from the kernel header files.  */ +#include <asm/ioctls.h> + +/* Oh well, this is necessary since the kernel data structure is +   different from the user-level version.  */ +#undef  TCGETS +#undef  TCSETS +#undef  TCSETSW +#undef  TCSETSF +#define TCGETS	_IOR ('T', 8, char[36]) +#define TCSETS	_IOW ('T', 9, char[36]) +#define TCSETSW	_IOW ('T', 10, char[36]) +#define TCSETSF	_IOW ('T', 11, char[36]) + +#include <linux/sockios.h> diff --git a/libc/sysdeps/linux/sparc64/bits/ipc.h b/libc/sysdeps/linux/sparc64/bits/ipc.h new file mode 100644 index 000000000..7fedafa61 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/ipc.h @@ -0,0 +1,53 @@ +/* Copyright (C) 1995-1999, 2000 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_IPC_H +# error "Never use <bits/ipc.h> directly; include <sys/ipc.h> instead." +#endif + +#include <bits/types.h> + +/* Mode bits for `msgget', `semget', and `shmget'.  */ +#define IPC_CREAT	01000		/* Create key if key does not exist. */ +#define IPC_EXCL	02000		/* Fail if key exists.  */ +#define IPC_NOWAIT	04000		/* Return error on wait.  */ + +/* Control commands for `msgctl', `semctl', and `shmctl'.  */ +#define IPC_RMID	0		/* Remove identifier.  */ +#define IPC_SET		1		/* Set `ipc_perm' options.  */ +#define IPC_STAT	2		/* Get `ipc_perm' options.  */ +#ifdef __USE_GNU +# define IPC_INFO	3		/* See ipcs.  */ +#endif + +/* Special key values.  */ +#define IPC_PRIVATE	((__key_t) 0)	/* Private key.  */ + + +/* Data structure used to pass permission information to IPC operations.  */ +struct ipc_perm +  { +    __key_t __key;			/* Key.  */ +    __uid_t uid;			/* Owner's user ID.  */ +    __gid_t gid;			/* Owner's group ID.  */ +    __uid_t cuid;			/* Creator's user ID.  */ +    __gid_t cgid;			/* Creator's group ID.  */ +    __mode_t mode;			/* Read/write permission.  */ +    unsigned short int __pad1; +    unsigned short int __seq;		/* Sequence number.  */ +    unsigned long long int __unused1; +    unsigned long long int __unused2; +  }; diff --git a/libc/sysdeps/linux/sparc64/bits/kernel_stat.h b/libc/sysdeps/linux/sparc64/bits/kernel_stat.h new file mode 100644 index 000000000..c45a595d5 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/kernel_stat.h @@ -0,0 +1,46 @@ +#ifndef _BITS_STAT_STRUCT_H +#define _BITS_STAT_STRUCT_H + +/* This file provides whatever this particular arch's kernel thinks + * struct kernel_stat should look like...  It turns out each arch has a + * different opinion on the subject... */ + +struct kernel_stat { +    unsigned int	st_dev; +    unsigned long	st_ino; +    unsigned short	st_mode; +    short		st_nlink; +    unsigned short	st_uid; +    unsigned short	st_gid; +    unsigned int	st_rdev; +    long		st_size; +    struct timespec	st_atim; +    struct timespec	st_mtim; +    struct timespec	st_ctim; +    long		st_blksize; +    long		st_blocks; +    unsigned long 	__unused4; +    unsigned long 	__unused5; +}; + +struct kernel_stat64 { +	unsigned long 	st_dev; +	unsigned long   st_ino; +	unsigned long	st_nlink; +	unsigned int	st_mode; +	unsigned int	st_uid; +	unsigned int	st_gid; +    	unsigned int 	__pad2; +	unsigned long	st_rdev; +	long 	st_size; +	long 	st_blksize; +	long 	st_blocks; +	struct timespec	st_atim; +	struct timespec	st_mtim; +	struct timespec	st_ctim; +	long	__unused4; +	long 	__unused5; +	long 	__unused6; +}; + +#endif	/*  _BITS_STAT_STRUCT_H */ diff --git a/libc/sysdeps/linux/sparc64/bits/kernel_types.h b/libc/sysdeps/linux/sparc64/bits/kernel_types.h new file mode 100644 index 000000000..d38ac8e40 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/kernel_types.h @@ -0,0 +1,43 @@ +/* Note that we use the exact same include guard #define names + * as asm/posix_types.h.  This will avoid gratuitous conflicts + * with the posix_types.h kernel header, and will ensure that + * our private content, and not the kernel header, will win. + *  -Erik + */ +#ifndef __ASM_GENERIC_POSIX_TYPES_H +#define __ASM_GENERIC_POSIX_TYPES_H + +typedef unsigned long          __kernel_size_t; +typedef long                   __kernel_ssize_t; +typedef long                   __kernel_ptrdiff_t; +typedef long                   __kernel_time_t; +typedef long                   __kernel_clock_t; +typedef int                    __kernel_pid_t; +typedef int                    __kernel_ipc_pid_t; +typedef unsigned int           __kernel_uid_t; +typedef unsigned int           __kernel_gid_t; +typedef unsigned int           __kernel_dev_t; +typedef unsigned long          __kernel_ino_t; +typedef unsigned int           __kernel_mode_t; +typedef unsigned short         __kernel_umode_t; +typedef unsigned int           __kernel_nlink_t; +typedef int                    __kernel_daddr_t; +typedef long                   __kernel_off_t; +typedef char *                 __kernel_caddr_t; +typedef unsigned short	       __kernel_uid16_t; +typedef unsigned short	       __kernel_gid16_t; +typedef unsigned short 	       __kernel_old_uid_t; +typedef unsigned short         __kernel_old_gid_t; +typedef __kernel_dev_t         __kernel_old_dev_t; +typedef long                   __kernel_long_t; +typedef unsigned long          __kernel_ulong_t; +typedef unsigned int	       __kernel_uid32_t; +typedef unsigned int	       __kernel_gid32_t; +typedef int		       __kernel_suseconds_t; +typedef long long              __kernel_loff_t; + +typedef struct { +	int val[2]; +} __kernel_fsid_t; + +#endif diff --git a/libc/sysdeps/linux/sparc64/bits/mathdef.h b/libc/sysdeps/linux/sparc64/bits/mathdef.h new file mode 100644 index 000000000..921a2bce5 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/mathdef.h @@ -0,0 +1,40 @@ +/* Copyright (C) 1997, 1998, 1999, 2000, 2004, 2006 +   Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#if !defined _MATH_H && !defined _COMPLEX_H +# error "Never use <bits/mathdef.h> directly; include <math.h> instead" +#endif + +/* FIXME! This file describes properties of the compiler, not the machine; +   it should not be part of libc!  */ + +#if defined __USE_ISOC99 && defined _MATH_H && !defined _MATH_H_MATHDEF +# define _MATH_H_MATHDEF	1 + +/* SPARC has both `float' and `double' arithmetic.  */ +typedef float float_t; +typedef double double_t; + +/* The values returned by `ilogb' for 0 and NaN respectively.  */ +# define FP_ILOGB0       (-2147483647) +# define FP_ILOGBNAN     (2147483647) + +#endif	/* ISO C99 */ + +#if !defined __NO_LONG_DOUBLE_MATH && !defined __UCLIBC_HAS_LONG_DOUBLE_MATH__ +# define __NO_LONG_DOUBLE_MATH	1 +#endif diff --git a/libc/sysdeps/linux/sparc64/bits/mathinline.h b/libc/sysdeps/linux/sparc64/bits/mathinline.h new file mode 100644 index 000000000..8bb35f6d8 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/mathinline.h @@ -0,0 +1,197 @@ +/* Inline math functions for SPARC. +   Copyright (C) 1999, 2000, 2001, 2002, 2004, 2006 +   Free Software Foundation, Inc. +   Contributed by Jakub Jelinek <jakub@redhat.com>. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _MATH_H +# error "Never use <bits/mathinline.h> directly; include <math.h> instead." +#endif + +#ifdef __GNUC__ + +#if defined __USE_ISOC99 && !__GNUC_PREREQ (3, 0) +# undef isgreater +# undef isgreaterequal +# undef isless +# undef islessequal +# undef islessgreater +# undef isunordered + +# define __unordered_v9cmp(x, y, op, qop) \ +  (__extension__							      \ +   ({ unsigned __r;							      \ +      if (sizeof (x) == 4 && sizeof (y) == 4)				      \ +	{								      \ +	  float __x = (x); float __y = (y);				      \ +	  __asm__ ("fcmps\t%%fcc3,%1,%2\n\tmov" op "\t%%fcc3,1,%0"	      \ +		   : "=r" (__r) : "f" (__x), "f" (__y), "0" (0) : "cc");      \ +	}								      \ +      else if (sizeof (x) <= 8 && sizeof (y) <= 8)			      \ +	{								      \ +	  double __x = (x); double __y = (y);				      \ +	  __asm__ ("fcmpd\t%%fcc3,%1,%2\n\tmov" op "\t%%fcc3,1,%0"	      \ +		   : "=r" (__r) : "f" (__x), "f" (__y), "0" (0) : "cc");      \ +	}								      \ +      else								      \ +	{								      \ +	  long double __x = (x); long double __y = (y);			      \ +	  extern int _Qp_cmp (const long double *a, const long double *b);    \ +	  __r = qop;							      \ +	}								      \ +      __r; })) + +# define isgreater(x, y) __unordered_v9cmp(x, y, "g", _Qp_cmp (&__x, &__y) == 2) +# define isgreaterequal(x, y) __unordered_v9cmp(x, y, "ge", (_Qp_cmp (&__x, &__y) & 1) == 0) +# define isless(x, y) __unordered_v9cmp(x, y, "l", _Qp_cmp (&__x, &__y) == 1) +# define islessequal(x, y) __unordered_v9cmp(x, y, "le", (_Qp_cmp (&__x, &__y) & 2) == 0) +# define islessgreater(x, y) __unordered_v9cmp(x, y, "lg", ((_Qp_cmp (&__x, &__y) + 1) & 2) != 0) +# define isunordered(x, y) __unordered_v9cmp(x, y, "u", _Qp_cmp (&__x, &__y) == 3) + +#endif /* __USE_ISOC99 */ + +#if (!defined __NO_MATH_INLINES || defined __LIBC_INTERNAL_MATH_INLINES) && defined __OPTIMIZE__ + +# ifdef __cplusplus +#  define __MATH_INLINE __inline +# else +#  define __MATH_INLINE __extern_inline +# endif  /* __cplusplus */ + +/* The gcc, version 2.7 or below, has problems with all this inlining +   code.  So disable it for this version of the compiler.  */ +# if __GNUC_PREREQ (2, 8) + +#  ifdef __USE_ISOC99 + +/* Test for negative number.  Used in the signbit() macro.  */ +__MATH_INLINE int +__NTH (__signbitf (float __x)) +{ +  __extension__ union { float __f; int __i; } __u = { __f: __x }; +  return __u.__i < 0; +} + +__MATH_INLINE int +__NTH (__signbit (double __x)) +{ +  __extension__ union { double __d; long int __i; } __u = { __d: __x }; +  return __u.__i < 0; +} + +__MATH_INLINE int +__NTH (__signbitl (long double __x)) +{ +  __extension__ union { long double __l; long int __i[2]; } __u = { __l: __x }; +  return __u.__i[0] < 0; +} + +#  endif /* __USE_ISOC99 */ + +#  if !defined __NO_MATH_INLINES && !__GNUC_PREREQ (3, 2) + +__MATH_INLINE double +__NTH (sqrt (double __x)) +{ +  register double __r; +  __asm__ ("fsqrtd %1,%0" : "=f" (__r) : "f" (__x)); +  return __r; +} + +__MATH_INLINE float +__NTH (sqrtf (float __x)) +{ +  register float __r; +  __asm__ ("fsqrts %1,%0" : "=f" (__r) : "f" (__x)); +  return __r; +} + +__MATH_INLINE long double +__NTH (sqrtl (long double __x)) +{ +  long double __r; +  extern void _Qp_sqrt (long double *, __const__ long double *); +  _Qp_sqrt (&__r, &__x); +  return __r; +} +#   elif !defined __NO_LONG_DOUBLE_MATH +__MATH_INLINE long double +sqrtl (long double __x) __THROW +{ +  extern long double _Q_sqrt (__const__ long double); +  return _Q_sqrt (__x); +} + +#  endif /* !__NO_MATH_INLINES && !GCC 3.2+ */ + +/* This code is used internally in the GNU libc.  */ +#  ifdef __LIBC_INTERNAL_MATH_INLINES +__MATH_INLINE double +__ieee754_sqrt (double __x) +{ +  register double __r; +  __asm__ ("fsqrtd %1,%0" : "=f" (__r) : "f" (__x)); +  return __r; +} + +__MATH_INLINE float +__ieee754_sqrtf (float __x) +{ +  register float __r; +  __asm__ ("fsqrts %1,%0" : "=f" (__r) : "f" (__x)); +  return __r; +} + +__MATH_INLINE long double +__ieee754_sqrtl (long double __x) +{ +  long double __r; +  extern void _Qp_sqrt (long double *, __const__ long double *); +  _Qp_sqrt(&__r, &__x); +  return __r; +} +#   elif !defined __NO_LONG_DOUBLE_MATH +__MATH_INLINE long double +__ieee754_sqrtl (long double __x) +{ +  extern long double _Q_sqrt (__const__ long double); +  return _Q_sqrt (__x); +} +#  endif /* __LIBC_INTERNAL_MATH_INLINES */ +# endif /* gcc 2.8+ */ + +# ifdef __USE_ISOC99 + +#  ifndef __NO_MATH_INLINES + +__MATH_INLINE double __NTH (fdim (double __x, double __y)); +__MATH_INLINE double +__NTH (fdim (double __x, double __y)) +{ +  return __x <= __y ? 0 : __x - __y; +} + +__MATH_INLINE float __NTH (fdimf (float __x, float __y)); +__MATH_INLINE float +__NTH (fdimf (float __x, float __y)) +{ +  return __x <= __y ? 0 : __x - __y; +} + +#  endif /* !__NO_MATH_INLINES */ +# endif /* __USE_ISOC99 */ +#endif /* !__NO_MATH_INLINES && __OPTIMIZE__ */ +#endif /* __GNUC__ */ diff --git a/libc/sysdeps/linux/sparc64/bits/mman.h b/libc/sysdeps/linux/sparc64/bits/mman.h new file mode 100644 index 000000000..9ac2b7011 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/mman.h @@ -0,0 +1,106 @@ +/* Definitions for POSIX memory map interface.  Linux/SPARC version. +   Copyright (C) 1997,1999,2000,2003,2005,2006 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_MMAN_H +# error "Never use <bits/mman.h> directly; include <sys/mman.h> instead." +#endif + +/* The following definitions basically come from the kernel headers. +   But the kernel header is not namespace clean.  */ + + +/* Protections are chosen from these bits, OR'd together.  The +   implementation does not necessarily support PROT_EXEC or PROT_WRITE +   without PROT_READ.  The only guarantees are that no writing will be +   allowed without PROT_WRITE and no access will be allowed for PROT_NONE. */ + +#define PROT_READ	0x1		/* Page can be read.  */ +#define PROT_WRITE	0x2		/* Page can be written.  */ +#define PROT_EXEC	0x4		/* Page can be executed.  */ +#define PROT_NONE	0x0		/* Page can not be accessed.  */ +#define PROT_GROWSDOWN	0x01000000	/* Extend change to start of +					   growsdown vma (mprotect only).  */ +#define PROT_GROWSUP	0x02000000	/* Extend change to start of +					   growsup vma (mprotect only).  */ + +/* Sharing types (must choose one and only one of these).  */ +#define MAP_SHARED	0x01		/* Share changes.  */ +#define MAP_PRIVATE	0x02		/* Changes are private.  */ +#ifdef __USE_MISC +# define MAP_TYPE	0x0f		/* Mask for type of mapping.  */ +#endif + +/* Other flags.  */ +#define MAP_FIXED	0x10		/* Interpret addr exactly.  */ +#ifdef __USE_MISC +# define MAP_FILE	0x00 +# define MAP_ANONYMOUS	0x20		/* Don't use a file.  */ +# define MAP_ANON	MAP_ANONYMOUS +# define MAP_RENAME	MAP_ANONYMOUS +#endif + +/* These are Linux-specific.  */ +#ifdef __USE_MISC +# define MAP_GROWSDOWN	0x0200		/* Stack-like segment.  */ +# define MAP_DENYWRITE	0x0800		/* ETXTBSY */ +# define MAP_EXECUTABLE	0x1000		/* Mark it as an executable.  */ +# define MAP_LOCKED	0x0100		/* Lock the mapping.  */ +# define MAP_NORESERVE	0x0040		/* Don't check for reservations.  */ +# define _MAP_NEW	0x80000000	/* Binary compatibility with SunOS.  */ +# define MAP_POPULATE	0x8000		/* Populate (prefault) pagetables.  */ +# define MAP_NONBLOCK	0x10000		/* Do not block on IO.  */ +# define MAP_UNINITIALIZED 0x4000000    /* For anonymous mmap, memory could +					   be uninitialized. */ +#endif + +/* Flags to `msync'.  */ +#define MS_ASYNC	1		/* Sync memory asynchronously.  */ +#define MS_SYNC		4		/* Synchronous memory sync.  */ +#define MS_INVALIDATE	2		/* Invalidate the caches.  */ + +/* Flags for `mlockall'.  */ +#define MCL_CURRENT	0x2000		/* Lock all currently mapped pages.  */ +#define MCL_FUTURE	0x4000		/* Lock all additions to address +					   space.  */ + +/* Flags for `mremap'.  */ +#ifdef __USE_GNU +# define MREMAP_MAYMOVE	1 +# define MREMAP_FIXED	2 +#endif + +/* Advice to `madvise'.  */ +#ifdef __USE_BSD +# define MADV_NORMAL	 0	/* No further special treatment.  */ +# define MADV_RANDOM	 1	/* Expect random page references.  */ +# define MADV_SEQUENTIAL 2	/* Expect sequential page references.  */ +# define MADV_WILLNEED	 3	/* Will need these pages.  */ +# define MADV_DONTNEED	 4	/* Don't need these pages.  */ +# define MADV_FREE	 5	/* Content can be freed (Solaris).  */ +# define MADV_REMOVE	 9	/* Remove these pages and resources.  */ +# define MADV_DONTFORK	 10	/* Do not inherit across fork.  */ +# define MADV_DOFORK	 11	/* Do inherit across fork.  */ +#endif + +/* The POSIX people had to invent similar names for the same things.  */ +#ifdef __USE_XOPEN2K +# define POSIX_MADV_NORMAL	0 /* No further special treatment.  */ +# define POSIX_MADV_RANDOM	1 /* Expect random page references.  */ +# define POSIX_MADV_SEQUENTIAL	2 /* Expect sequential page references.  */ +# define POSIX_MADV_WILLNEED	3 /* Will need these pages.  */ +# define POSIX_MADV_DONTNEED	4 /* Don't need these pages.  */ +#endif diff --git a/libc/sysdeps/linux/sparc64/bits/msq.h b/libc/sysdeps/linux/sparc64/bits/msq.h new file mode 100644 index 000000000..f514e442c --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/msq.h @@ -0,0 +1,72 @@ +/* Copyright (C) 1995, 1996, 1997, 2000 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_MSG_H +# error "Never use <bits/msq.h> directly; include <sys/msg.h> instead." +#endif + +#include <bits/types.h> + +/* Define options for message queue functions.  */ +#define MSG_NOERROR	010000	/* no error if message is too big */ +#ifdef __USE_GNU +# define MSG_EXCEPT	020000	/* recv any msg except of specified type */ +#endif + +/* Types used in the structure definition.  */ +typedef unsigned long int msgqnum_t; +typedef unsigned long int msglen_t; + + +/* Structure of record for one message inside the kernel. +   The type `struct msg' is opaque.  */ +struct msqid_ds +{ +  struct ipc_perm msg_perm;	/* structure describing operation permission */ +  __time_t msg_stime;		/* time of last msgsnd command */ +  __time_t msg_rtime;		/* time of last msgrcv command */ +  __time_t msg_ctime;		/* time of last change */ +  unsigned long int __msg_cbytes; /* current number of bytes on queue */ +  msgqnum_t msg_qnum;		/* number of messages currently on queue */ +  msglen_t msg_qbytes;		/* max number of bytes allowed on queue */ +  __pid_t msg_lspid;		/* pid of last msgsnd() */ +  __pid_t msg_lrpid;		/* pid of last msgrcv() */ +  unsigned long int __unused1; +  unsigned long int __unused2; +}; + +#ifdef __USE_MISC + +# define msg_cbytes	__msg_cbytes + +/* ipcs ctl commands */ +# define MSG_STAT 11 +# define MSG_INFO 12 + +/* buffer for msgctl calls IPC_INFO, MSG_INFO */ +struct msginfo +  { +    int msgpool; +    int msgmap; +    int msgmax; +    int msgmnb; +    int msgmni; +    int msgssz; +    int msgtql; +    unsigned short int msgseg; +  }; + +#endif /* __USE_MISC */ diff --git a/libc/sysdeps/linux/sparc64/bits/poll.h b/libc/sysdeps/linux/sparc64/bits/poll.h new file mode 100644 index 000000000..b6ebbb4c4 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/poll.h @@ -0,0 +1,48 @@ +/* Copyright (C) 1997, 2001, 2006 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_POLL_H +# error "Never use <bits/poll.h> directly; include <sys/poll.h> instead." +#endif + +/* Event types that can be polled for.  These bits may be set in `events' +   to indicate the interesting event types; they will appear in `revents' +   to indicate the status of the file descriptor.  */ +#define POLLIN		0x001		/* There is data to read.  */ +#define POLLPRI		0x002		/* There is urgent data to read.  */ +#define POLLOUT		0x004		/* Writing now will not block.  */ + +#ifdef __USE_XOPEN +/* These values are defined in XPG4.2.  */ +# define POLLRDNORM	0x040		/* Normal data may be read.  */ +# define POLLRDBAND	0x080		/* Priority data may be read.  */ +# define POLLWRNORM	POLLOUT		/* Writing now will not block.  */ +# define POLLWRBAND	0x100		/* Priority data may be written.  */ +#endif + +#ifdef __USE_GNU +/* These are extensions for Linux.  */ +# define POLLMSG	0x200 +# define POLLREMOVE	0x400 +# define POLLRDHUP	0x800 +#endif + +/* Event types always implicitly polled for.  These bits need not be set in +   `events', but they will appear in `revents' to indicate the status of +   the file descriptor.  */ +#define POLLERR		0x008		/* Error condition.  */ +#define POLLHUP		0x010		/* Hung up.  */ +#define POLLNVAL	0x020		/* Invalid polling request.  */ diff --git a/libc/sysdeps/linux/sparc64/bits/resource.h b/libc/sysdeps/linux/sparc64/bits/resource.h new file mode 100644 index 000000000..366e5c2d9 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/resource.h @@ -0,0 +1,239 @@ +/* Bit values & structures for resource limits.  Linux/SPARC version. +   Copyright (C) 1994, 1996, 1997, 1998, 1999, 2000, 2004, 2005 +   Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_RESOURCE_H +# error "Never use <bits/resource.h> directly; include <sys/resource.h> instead." +#endif + +#include <bits/types.h> + +/* Transmute defines to enumerations.  The macro re-definitions are +   necessary because some programs want to test for operating system +   features with #ifdef RUSAGE_SELF.  In ISO C the reflexive +   definition is a no-op.  */ + +/* Kinds of resource limit.  */ +enum __rlimit_resource +{ +  /* Per-process CPU limit, in seconds.  */ +  RLIMIT_CPU = 0, +#define RLIMIT_CPU RLIMIT_CPU + +  /* Largest file that can be created, in bytes.  */ +  RLIMIT_FSIZE = 1, +#define	RLIMIT_FSIZE RLIMIT_FSIZE + +  /* Maximum size of data segment, in bytes.  */ +  RLIMIT_DATA = 2, +#define	RLIMIT_DATA RLIMIT_DATA + +  /* Maximum size of stack segment, in bytes.  */ +  RLIMIT_STACK = 3, +#define	RLIMIT_STACK RLIMIT_STACK + +  /* Largest core file that can be created, in bytes.  */ +  RLIMIT_CORE = 4, +#define	RLIMIT_CORE RLIMIT_CORE + +  /* Largest resident set size, in bytes. +     This affects swapping; processes that are exceeding their +     resident set size will be more likely to have physical memory +     taken from them.  */ +  __RLIMIT_RSS = 5, +#define	RLIMIT_RSS __RLIMIT_RSS + +  /* Number of open files.  */ +  RLIMIT_NOFILE = 6, +  __RLIMIT_OFILE = RLIMIT_NOFILE, /* BSD name for same.  */ +#define RLIMIT_NOFILE RLIMIT_NOFILE +#define RLIMIT_OFILE __RLIMIT_OFILE + +  /* Address space limit (?) */ +  RLIMIT_AS = 9, +#define RLIMIT_AS RLIMIT_AS + +  /* Number of processes.  */ +  __RLIMIT_NPROC = 7, +#define RLIMIT_NPROC __RLIMIT_NPROC + +  /* Locked-in-memory address space.  */ +  __RLIMIT_MEMLOCK = 8, +#define RLIMIT_MEMLOCK __RLIMIT_MEMLOCK + +  /* Maximum number of file locks.  */ +  __RLIMIT_LOCKS = 10, +#define RLIMIT_LOCKS __RLIMIT_LOCKS + +  /* Maximum number of pending signals.  */ +  __RLIMIT_SIGPENDING = 11, +#define RLIMIT_SIGPENDING __RLIMIT_SIGPENDING + +  /* Maximum bytes in POSIX message queues.  */ +  __RLIMIT_MSGQUEUE = 12, +#define RLIMIT_MSGQUEUE __RLIMIT_MSGQUEUE + +  /* Maximum nice priority allowed to raise to. +     Nice levels 19 .. -20 correspond to 0 .. 39 +     values of this resource limit.  */ +  __RLIMIT_NICE = 13, +#define RLIMIT_NICE __RLIMIT_NICE + +  /* Maximum realtime priority allowed for non-priviledged +     processes.  */ +  __RLIMIT_RTPRIO = 14, +#define RLIMIT_RTPRIO __RLIMIT_RTPRIO + +  __RLIMIT_NLIMITS = 15, +  __RLIM_NLIMITS = __RLIMIT_NLIMITS +#define RLIMIT_NLIMITS __RLIMIT_NLIMITS +#define RLIM_NLIMITS __RLIM_NLIMITS +}; + +/* Value to indicate that there is no limit.  */ +#if __WORDSIZE == 64 + +#ifndef __USE_FILE_OFFSET64 +# define RLIM_INFINITY ((unsigned long int)(~0UL)) +#else +# define RLIM_INFINITY 0xffffffffffffffffuLL +#endif + +#ifdef __USE_LARGEFILE64 +# define RLIM64_INFINITY 0xffffffffffffffffuLL +#endif + +#else + +#ifndef __USE_FILE_OFFSET64 +# define RLIM_INFINITY ((long int)(~0UL >> 1)) +#else +# define RLIM_INFINITY 0x7fffffffffffffffLL +#endif + +#ifdef __USE_LARGEFILE64 +# define RLIM64_INFINITY 0x7fffffffffffffffLL +#endif + +#endif + +/* We can represent all limits.  */ +#define RLIM_SAVED_MAX	RLIM_INFINITY +#define RLIM_SAVED_CUR	RLIM_INFINITY + + +/* Type for resource quantity measurement.  */ +#ifndef __USE_FILE_OFFSET64 +typedef __rlim_t rlim_t; +#else +typedef __rlim64_t rlim_t; +#endif +#ifdef __USE_LARGEFILE64 +typedef __rlim64_t rlim64_t; +#endif + +struct rlimit +  { +    /* The current (soft) limit.  */ +    rlim_t rlim_cur; +    /* The hard limit.  */ +    rlim_t rlim_max; +  }; + +#ifdef __USE_LARGEFILE64 +struct rlimit64 +  { +    /* The current (soft) limit.  */ +    rlim64_t rlim_cur; +    /* The hard limit.  */ +    rlim64_t rlim_max; + }; +#endif + +/* Whose usage statistics do you want?  */ +enum __rusage_who +{ +  /* The calling process.  */ +  RUSAGE_SELF = 0, +#define RUSAGE_SELF RUSAGE_SELF + +  /* All of its terminated child processes.  */ +  RUSAGE_CHILDREN = -1 +#define RUSAGE_CHILDREN RUSAGE_CHILDREN +}; + +#define __need_timeval +#include <bits/time.h>		/* For `struct timeval'.  */ + +/* Structure which says how much of each resource has been used.  */ +struct rusage +  { +    /* Total amount of user time used.  */ +    struct timeval ru_utime; +    /* Total amount of system time used.  */ +    struct timeval ru_stime; +    /* Maximum resident set size (in kilobytes).  */ +    long int ru_maxrss; +    /* Amount of sharing of text segment memory +       with other processes (kilobyte-seconds).  */ +    long int ru_ixrss; +    /* Amount of data segment memory used (kilobyte-seconds).  */ +    long int ru_idrss; +    /* Amount of stack memory used (kilobyte-seconds).  */ +    long int ru_isrss; +    /* Number of soft page faults (i.e. those serviced by reclaiming +       a page from the list of pages awaiting reallocation.  */ +    long int ru_minflt; +    /* Number of hard page faults (i.e. those that required I/O).  */ +    long int ru_majflt; +    /* Number of times a process was swapped out of physical memory.  */ +    long int ru_nswap; +    /* Number of input operations via the file system.  Note: This +       and `ru_oublock' do not include operations with the cache.  */ +    long int ru_inblock; +    /* Number of output operations via the file system.  */ +    long int ru_oublock; +    /* Number of IPC messages sent.  */ +    long int ru_msgsnd; +    /* Number of IPC messages received.  */ +    long int ru_msgrcv; +    /* Number of signals delivered.  */ +    long int ru_nsignals; +    /* Number of voluntary context switches, i.e. because the process +       gave up the process before it had to (usually to wait for some +       resource to be available).  */ +    long int ru_nvcsw; +    /* Number of involuntary context switches, i.e. a higher priority process +       became runnable or the current process used up its time slice.  */ +    long int ru_nivcsw; +  }; + +/* Priority limits.  */ +#define PRIO_MIN	-20	/* Minimum priority a process can have.  */ +#define PRIO_MAX	20	/* Maximum priority a process can have.  */ + +/* The type of the WHICH argument to `getpriority' and `setpriority', +   indicating what flavor of entity the WHO argument specifies.  */ +enum __priority_which +{ +  PRIO_PROCESS = 0,		/* WHO is a process ID.  */ +#define PRIO_PROCESS PRIO_PROCESS +  PRIO_PGRP = 1,		/* WHO is a process group ID.  */ +#define PRIO_PGRP PRIO_PGRP +  PRIO_USER = 2			/* WHO is a user ID.  */ +#define PRIO_USER PRIO_USER +}; diff --git a/libc/sysdeps/linux/sparc64/bits/sem.h b/libc/sysdeps/linux/sparc64/bits/sem.h new file mode 100644 index 000000000..acc21742d --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/sem.h @@ -0,0 +1,83 @@ +/* Copyright (C) 1995, 1996, 1997, 1998, 2000 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_SEM_H +# error "Never include <bits/sem.h> directly; use <sys/sem.h> instead." +#endif + +#include <sys/types.h> + +/* Flags for `semop'.  */ +#define SEM_UNDO	0x1000		/* undo the operation on exit */ + +/* Commands for `semctl'.  */ +#define GETPID		11		/* get sempid */ +#define GETVAL		12		/* get semval */ +#define GETALL		13		/* get all semval's */ +#define GETNCNT		14		/* get semncnt */ +#define GETZCNT		15		/* get semzcnt */ +#define SETVAL		16		/* set semval */ +#define SETALL		17		/* set all semval's */ + + +/* Data structure describing a set of semaphores.  */ +struct semid_ds +{ +  struct ipc_perm sem_perm;		/* operation permission struct */ +  __time_t sem_otime;			/* last semop() time */ +  __time_t sem_ctime;			/* last time changed by semctl() */ +  unsigned long int sem_nsems;		/* number of semaphores in set */ +  unsigned long int __unused1; +  unsigned long int __unused2; +}; + +/* The user should define a union like the following to use it for arguments +   for `semctl'. + +   union semun +   { +     int val;				<= value for SETVAL +     struct semid_ds *buf;		<= buffer for IPC_STAT & IPC_SET +     unsigned short int *array;		<= array for GETALL & SETALL +     struct seminfo *__buf;		<= buffer for IPC_INFO +   }; + +   Previous versions of this file used to define this union but this is +   incorrect.  One can test the macro _SEM_SEMUN_UNDEFINED to see whether +   one must define the union or not.  */ +#define _SEM_SEMUN_UNDEFINED	1 + +#ifdef __USE_MISC + +/* ipcs ctl cmds */ +# define SEM_STAT 18 +# define SEM_INFO 19 + +struct  seminfo +{ +  int semmap; +  int semmni; +  int semmns; +  int semmnu; +  int semmsl; +  int semopm; +  int semume; +  int semusz; +  int semvmx; +  int semaem; +}; + +#endif /* __USE_MISC */ diff --git a/libc/sysdeps/linux/sparc64/bits/setjmp.h b/libc/sysdeps/linux/sparc64/bits/setjmp.h new file mode 100644 index 000000000..7cca64f82 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/setjmp.h @@ -0,0 +1,54 @@ +/* Copyright (C) 1997-2017 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _BITS_SETJMP_H +#define _BITS_SETJMP_H  1 + +#if !defined _SETJMP_H && !defined _PTHREAD_H +# error "Never include <bits/setjmp.h> directly; use <setjmp.h> instead." +#endif + +#ifndef _ASM +typedef struct __sparc64_jmp_buf +  { +    struct __sparc64_jmp_buf	*__uc_link; +    unsigned long		__uc_flags; +    unsigned long		__uc_sigmask; +    struct __sparc64_jmp_buf_mcontext +      { +	unsigned long		__mc_gregs[19]; +	unsigned long		__mc_fp; +	unsigned long		__mc_i7; +	struct __sparc64_jmp_buf_fpu +	  { +	    union +	      { +		unsigned int	__sregs[32]; +		unsigned long	__dregs[32]; +		long double	__qregs[16]; +	      }			__mcfpu_fpregs; +	    unsigned long	__mcfpu_fprs; +	    unsigned long	__mcfpu_gsr; +	    void		*__mcfpu_fq; +	    unsigned char	__mcfpu_qcnt; +	    unsigned char	__mcfpu_qentsz; +	    unsigned char	__mcfpu_enab; +	  }			__mc_fpregs; +      }				__uc_mcontext; +  } __jmp_buf[1]; +#endif + +#endif  /* bits/setjmp.h */ diff --git a/libc/sysdeps/linux/sparc64/bits/shm.h b/libc/sysdeps/linux/sparc64/bits/shm.h new file mode 100644 index 000000000..a5d8c3464 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/shm.h @@ -0,0 +1,99 @@ +/* Copyright (C) 1995, 1996, 1997, 2000, 2002, 2004 +   Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_SHM_H +# error "Never include <bits/shm.h> directly; use <sys/shm.h> instead." +#endif + +#include <bits/types.h> + +/* Permission flag for shmget.  */ +#define SHM_R		0400		/* or S_IRUGO from <linux/stat.h> */ +#define SHM_W		0200		/* or S_IWUGO from <linux/stat.h> */ + +/* Flags for `shmat'.  */ +#define SHM_RDONLY	010000		/* attach read-only else read-write */ +#define SHM_RND		020000		/* round attach address to SHMLBA */ +#define SHM_REMAP	040000		/* take-over region on attach */ + +/* Commands for `shmctl'.  */ +#define SHM_LOCK	11		/* lock segment (root only) */ +#define SHM_UNLOCK	12		/* unlock segment (root only) */ + +__BEGIN_DECLS + +/* Segment low boundary address multiple.  */ +#define SHMLBA		(__getpagesize ()) +extern int __getpagesize (void) __THROW __attribute__ ((__const__)); + + +/* Type to count number of attaches.  */ +typedef unsigned long int shmatt_t; + +/* Data structure describing a set of semaphores.  */ +struct shmid_ds +  { +    struct ipc_perm shm_perm;		/* operation permission struct */ +    __time_t shm_atime;			/* time of last shmat() */ +    __time_t shm_dtime;			/* time of last shmdt() */ +    __time_t shm_ctime;			/* time of last change by shmctl() */ +    size_t shm_segsz;			/* size of segment in bytes */ +    __pid_t shm_cpid;			/* pid of creator */ +    __pid_t shm_lpid;			/* pid of last shmop */ +    shmatt_t shm_nattch;		/* number of current attaches */ +    unsigned long int __unused1; +    unsigned long int __unused2; +  }; + +#ifdef __USE_MISC + +/* ipcs ctl commands */ +# define SHM_STAT 	13 +# define SHM_INFO 	14 + +/* shm_mode upper byte flags */ +# define SHM_DEST	01000	/* segment will be destroyed on last detach */ +# define SHM_LOCKED	02000   /* segment will not be swapped */ +# define SHM_HUGETLB	04000	/* segment is mapped via hugetlb */ +# define SHM_NORESERVE	010000	/* don't check for reservations */ + +struct	shminfo +  { +    unsigned long shmmax; +    unsigned long shmmin; +    unsigned long shmmni; +    unsigned long shmseg; +    unsigned long shmall; +    unsigned long __unused1; +    unsigned long __unused2; +    unsigned long __unused3; +    unsigned long __unused4; +  }; + +struct shm_info +  { +    int used_ids; +    unsigned long int shm_tot;	/* total allocated shm */ +    unsigned long int shm_rss;	/* total resident shm */ +    unsigned long int shm_swp;	/* total swapped shm */ +    unsigned long int swap_attempts; +    unsigned long int swap_successes; +  }; + +#endif /* __USE_MISC */ + +__END_DECLS diff --git a/libc/sysdeps/linux/sparc64/bits/sigaction.h b/libc/sysdeps/linux/sparc64/bits/sigaction.h new file mode 100644 index 000000000..6c5b735be --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/sigaction.h @@ -0,0 +1,77 @@ +/* The proper definitions for Linux/SPARC sigaction. +   Copyright (C) 1996-2017 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SIGNAL_H +# error "Never include <bits/sigaction.h> directly; use <signal.h> instead." +#endif + +/* Structure describing the action to be taken when a signal arrives.  */ +struct sigaction +  { +    /* Signal handler. */ +#ifdef __USE_POSIX199309 +    union +      { +	/* Used if SA_SIGINFO is not set.  */ +	__sighandler_t sa_handler; +	/* Used if SA_SIGINFO is set.  */ +	void (*sa_sigaction) (int, siginfo_t *, void *); +      } +    __sigaction_handler; +# define sa_handler	__sigaction_handler.sa_handler +# define sa_sigaction	__sigaction_handler.sa_sigaction +#else +    __sighandler_t sa_handler; +#endif + +    /* Additional set of signals to be blocked.  */ +    __sigset_t sa_mask; + +    /* Special flags.  */ +    int __glibc_reserved0; +    int sa_flags; + +    /* Not used by Linux/Sparc yet.  */ +    void (*sa_restorer) (void); +  }; + + +/* Bits in `sa_flags'.  */ +#define	SA_NOCLDSTOP 0x00000008  /* Don't send SIGCHLD when children stop.  */ +#define SA_NOCLDWAIT 0x00000100  /* Don't create zombie on child death.  */ +#define SA_SIGINFO   0x00000200  /* Invoke signal-catching function with +				    three arguments instead of one.  */ +#if defined __USE_UNIX98 || defined __USE_MISC +# define SA_ONSTACK   0x00000001 /* Use signal stack by using `sa_restorer'. */ +#endif +#if defined __USE_UNIX98 || defined __USE_XOPEN2K8 +# define SA_RESTART   0x00000002 /* Restart syscall on signal return.  */ +# define SA_INTERRUPT 0x00000010 /* Historical no-op.  */ +# define SA_NOMASK    0x00000020 /* Don't automatically block the signal when +				    its handler is being executed.  */ +# define SA_ONESHOT   0x00000004 /* Reset to SIG_DFL on entry to handler.  */ + +/* Some aliases for the SA_ constants.  */ +# define SA_NODEFER   SA_NOMASK +# define SA_RESETHAND SA_ONESHOT +# define SA_STACK     SA_ONSTACK +#endif + +/* Values for the HOW argument to `sigprocmask'.  */ +#define	SIG_BLOCK     1		 /* Block signals.  */ +#define	SIG_UNBLOCK   2		 /* Unblock signals.  */ +#define	SIG_SETMASK   4		 /* Set the set of blocked signals.  */ diff --git a/libc/sysdeps/linux/sparc64/bits/sigcontextinfo.h b/libc/sysdeps/linux/sparc64/bits/sigcontextinfo.h new file mode 100644 index 000000000..8e32a3e32 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/sigcontextinfo.h @@ -0,0 +1,30 @@ +/* Copyright (C) 1999-2017 Free Software Foundation, Inc. +   Contributed by Jakub Jelinek <jj@ultra.linux.cz>, 1999. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef STACK_BIAS +#define STACK_BIAS 2047 +#endif +#define SIGCONTEXT struct sigcontext * +#define SIGCONTEXT_EXTRA_ARGS +#define GET_PC(__ctx)	((void *) ((__ctx)->sigc_regs.tpc)) +#define ADVANCE_STACK_FRAME(__next) \ +	((void *) (((unsigned long *) (((unsigned long int) (__next))     \ +					   + STACK_BIAS))+14)) +#define GET_STACK(__ctx)	((void *) ((__ctx)->sigc_regs.u_regs[14])) +#define GET_FRAME(__ctx)	ADVANCE_STACK_FRAME (GET_STACK (__ctx)) +#define CALL_SIGHANDLER(handler, signo, ctx) \ +  (handler)((signo), SIGCONTEXT_EXTRA_ARGS (ctx)) diff --git a/libc/sysdeps/linux/sparc64/bits/siginfo.h b/libc/sysdeps/linux/sparc64/bits/siginfo.h new file mode 100644 index 000000000..d05a9e4b9 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/siginfo.h @@ -0,0 +1,321 @@ +/* siginfo_t, sigevent and constants.  Linux/SPARC version. +   Copyright (C) 1997-2002, 2003 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#if !defined _SIGNAL_H && !defined __need_siginfo_t \ +    && !defined __need_sigevent_t +# error "Never include this file directly.  Use <signal.h> instead" +#endif + +#if (!defined __have_sigval_t \ +     && (defined _SIGNAL_H || defined __need_siginfo_t \ +	 || defined __need_sigevent_t)) +# define __have_sigval_t	1 + +/* Type for data associated with a signal.  */ +typedef union sigval +  { +    int sival_int; +    void *sival_ptr; +  } sigval_t; +#endif + +#if (!defined __have_siginfo_t \ +     && (defined _SIGNAL_H || defined __need_siginfo_t)) +# define __have_siginfo_t	1 + +# define __SI_MAX_SIZE     128 +# define __SI_PAD_SIZE     ((__SI_MAX_SIZE / sizeof (int)) - 4) + +typedef struct siginfo +  { +    int si_signo;		/* Signal number.  */ +    int si_errno;		/* If non-zero, an errno value associated with +				   this signal, as defined in <errno.h>.  */ +    int si_code;		/* Signal code.  */ + +    union +      { +	int _pad[__SI_PAD_SIZE]; + +	 /* kill().  */ +	struct +	  { +	    __pid_t si_pid;	/* Sending process ID.  */ +	    __uid_t si_uid;	/* Real user ID of sending process.  */ +	  } _kill; + +	/* POSIX.1b timers.  */ +	struct +	  { +	    int si_tid;		/* Timer ID.  */ +	    int si_overrun;	/* Overrun count.  */ +	    sigval_t si_sigval;	/* Signal value.  */ +	  } _timer; + +	/* POSIX.1b signals.  */ +	struct +	  { +	    __pid_t si_pid;	/* Sending process ID.  */ +	    __uid_t si_uid;	/* Real user ID of sending process.  */ +	    sigval_t si_sigval;	/* Signal value.  */ +	  } _rt; + +	/* SIGCHLD.  */ +	struct +	  { +	    __pid_t si_pid;	/* Which child.  */ +	    __uid_t si_uid;	/* Real user ID of sending process.  */ +	    int si_status;	/* Exit value or signal.  */ +	    __clock_t si_utime; +	    __clock_t si_stime; +	  } _sigchld; + +	/* SIGILL, SIGFPE, SIGSEGV, SIGBUS.  */ +	struct +	  { +	    void *si_addr;	/* Faulting insn/memory ref.  */ +	    int si_trapno; +	  } _sigfault; + +	/* SIGPOLL.  */ +	struct +	  { +	    int si_band;	/* Band event for SIGPOLL.  */ +	    int si_fd; +	  } _sigpoll; + +	/* SIGSYS.  */ +	struct +	  { +	    void *_call_addr;   /* Calling user insn.  */ +	    int _syscall;       /* Triggering system call number.  */ +	    unsigned int _arch; /* AUDIT_ARCH_* of syscall.  */ +	  } _sigsys; +      } _sifields; +  } siginfo_t; + + +/* X/Open requires some more fields with fixed names.  */ +# define si_pid		_sifields._kill.si_pid +# define si_uid		_sifields._kill.si_uid +# define si_timerid	_sifields._timer.si_tid +# define si_overrun	_sifields._timer.si_overrun +# define si_status	_sifields._sigchld.si_status +# define si_utime	_sifields._sigchld.si_utime +# define si_stime	_sifields._sigchld.si_stime +# define si_value	_sifields._rt.si_sigval +# define si_int		_sifields._rt.si_sigval.sival_int +# define si_ptr		_sifields._rt.si_sigval.sival_ptr +# define si_addr	_sifields._sigfault.si_addr +# define si_trapno	_sifields._sigfault.si_trapno +# define si_band	_sifields._sigpoll.si_band +# define si_fd		_sifields._sigpoll.si_fd +# define si_call_addr	_sifields._sigsys._call_addr +# define si_syscall	_sifields._sigsys._syscall +# define si_arch	_sifields._sigsys._arch + + +/* Values for `si_code'.  Positive values are reserved for kernel-generated +   signals.  */ +enum +{ +  SI_ASYNCNL = -60,		/* Sent by asynch name lookup completion.  */ +# define SI_ASYNCNL	SI_ASYNCNL +  SI_TKILL = -6,		/* Sent by tkill.  */ +# define SI_TKILL	SI_TKILL +  SI_SIGIO,			/* Sent by queued SIGIO. */ +# define SI_SIGIO	SI_SIGIO +  SI_ASYNCIO,			/* Sent by AIO completion.  */ +# define SI_ASYNCIO	SI_ASYNCIO +  SI_MESGQ,			/* Sent by real time mesq state change.  */ +# define SI_MESGQ	SI_MESGQ +  SI_TIMER,			/* Sent by timer expiration.  */ +# define SI_TIMER	SI_TIMER +  SI_QUEUE,			/* Sent by sigqueue.  */ +# define SI_QUEUE	SI_QUEUE +  SI_USER,			/* Sent by kill, sigsend, raise.  */ +# define SI_USER	SI_USER +  SI_KERNEL = 0x80		/* Send by kernel.  */ +#define SI_KERNEL	SI_KERNEL +}; + + +/* `si_code' values for SIGILL signal.  */ +enum +{ +  ILL_ILLOPC = 1,		/* Illegal opcode.  */ +# define ILL_ILLOPC	ILL_ILLOPC +  ILL_ILLOPN,			/* Illegal operand.  */ +# define ILL_ILLOPN	ILL_ILLOPN +  ILL_ILLADR,			/* Illegal addressing mode.  */ +# define ILL_ILLADR	ILL_ILLADR +  ILL_ILLTRP,			/* Illegal trap. */ +# define ILL_ILLTRP	ILL_ILLTRP +  ILL_PRVOPC,			/* Privileged opcode.  */ +# define ILL_PRVOPC	ILL_PRVOPC +  ILL_PRVREG,			/* Privileged register.  */ +# define ILL_PRVREG	ILL_PRVREG +  ILL_COPROC,			/* Coprocessor error.  */ +# define ILL_COPROC	ILL_COPROC +  ILL_BADSTK			/* Internal stack error.  */ +# define ILL_BADSTK	ILL_BADSTK +}; + +/* `si_code' values for SIGFPE signal.  */ +enum +{ +  FPE_INTDIV = 1,		/* Integer divide by zero.  */ +# define FPE_INTDIV	FPE_INTDIV +  FPE_INTOVF,			/* Integer overflow.  */ +# define FPE_INTOVF	FPE_INTOVF +  FPE_FLTDIV,			/* Floating point divide by zero.  */ +# define FPE_FLTDIV	FPE_FLTDIV +  FPE_FLTOVF,			/* Floating point overflow.  */ +# define FPE_FLTOVF	FPE_FLTOVF +  FPE_FLTUND,			/* Floating point underflow.  */ +# define FPE_FLTUND	FPE_FLTUND +  FPE_FLTRES,			/* Floating point inexact result.  */ +# define FPE_FLTRES	FPE_FLTRES +  FPE_FLTINV,			/* Floating point invalid operation.  */ +# define FPE_FLTINV	FPE_FLTINV +  FPE_FLTSUB			/* Subscript out of range.  */ +# define FPE_FLTSUB	FPE_FLTSUB +}; + +/* `si_code' values for SIGSEGV signal.  */ +enum +{ +  SEGV_MAPERR = 1,		/* Address not mapped to object.  */ +# define SEGV_MAPERR	SEGV_MAPERR +  SEGV_ACCERR			/* Invalid permissions for mapped object.  */ +# define SEGV_ACCERR	SEGV_ACCERR +}; + +/* `si_code' values for SIGBUS signal.  */ +enum +{ +  BUS_ADRALN = 1,		/* Invalid address alignment.  */ +# define BUS_ADRALN	BUS_ADRALN +  BUS_ADRERR,			/* Non-existant physical address.  */ +# define BUS_ADRERR	BUS_ADRERR +  BUS_OBJERR			/* Object specific hardware error.  */ +# define BUS_OBJERR	BUS_OBJERR +}; + +/* `si_code' values for SIGTRAP signal.  */ +enum +{ +  TRAP_BRKPT = 1,		/* Process breakpoint.  */ +# define TRAP_BRKPT	TRAP_BRKPT +  TRAP_TRACE			/* Process trace trap.  */ +# define TRAP_TRACE	TRAP_TRACE +}; + +/* `si_code' values for SIGCHLD signal.  */ +enum +{ +  CLD_EXITED = 1,		/* Child has exited.  */ +# define CLD_EXITED	CLD_EXITED +  CLD_KILLED,			/* Child was killed.  */ +# define CLD_KILLED	CLD_KILLED +  CLD_DUMPED,			/* Child terminated abnormally.  */ +# define CLD_DUMPED	CLD_DUMPED +  CLD_TRAPPED,			/* Traced child has trapped.  */ +# define CLD_TRAPPED	CLD_TRAPPED +  CLD_STOPPED,			/* Child has stopped.  */ +# define CLD_STOPPED	CLD_STOPPED +  CLD_CONTINUED			/* Stopped child has continued.  */ +# define CLD_CONTINUED	CLD_CONTINUED +}; + +/* `si_code' values for SIGPOLL signal.  */ +enum +{ +  POLL_IN = 1,			/* Data input available.  */ +# define POLL_IN	POLL_IN +  POLL_OUT,			/* Output buffers available.  */ +# define POLL_OUT	POLL_OUT +  POLL_MSG,			/* Input message available.   */ +# define POLL_MSG	POLL_MSG +  POLL_ERR,			/* I/O error.  */ +# define POLL_ERR	POLL_ERR +  POLL_PRI,			/* High priority input available.  */ +# define POLL_PRI	POLL_PRI +  POLL_HUP			/* Device disconnected.  */ +# define POLL_HUP	POLL_HUP +}; + +/* `si_code' values for SIGEMT signal.  */ +enum +{ +  EMT_TAGOVF = 1		/* Tag overflow.  */ +# define EMT_TAGOVF	EMT_TAGOVF +}; + +# undef __need_siginfo_t +#endif	/* !have siginfo_t && (have _SIGNAL_H || need siginfo_t).  */ + + +#if (defined _SIGNAL_H || defined __need_sigevent_t) \ +    && !defined __have_sigevent_t +# define __have_sigevent_t	1 + +/* Structure to transport application-defined values with signals.  */ +# define __SIGEV_MAX_SIZE	64 +# define __SIGEV_PAD_SIZE	((__SIGEV_MAX_SIZE / sizeof (int)) - 4) + +typedef struct sigevent +  { +    sigval_t sigev_value; +    int sigev_signo; +    int sigev_notify; + +    union +      { +	int _pad[__SIGEV_PAD_SIZE]; + +	/* When SIGEV_SIGNAL and SIGEV_THREAD_ID set, LWP ID of the +	   thread to receive the signal.  */ +	__pid_t _tid; + +	struct +	  { +	    void (*_function) (sigval_t);	/* Function to start.  */ +	    void *_attribute;			/* Really pthread_attr_t.  */ +	  } _sigev_thread; +      } _sigev_un; +  } sigevent_t; + +/* POSIX names to access some of the members.  */ +# define sigev_notify_function   _sigev_un._sigev_thread._function +# define sigev_notify_attributes _sigev_un._sigev_thread._attribute + +/* `sigev_notify' values.  */ +enum +{ +  SIGEV_SIGNAL = 0,		/* Notify via signal.  */ +# define SIGEV_SIGNAL	SIGEV_SIGNAL +  SIGEV_NONE,			/* Other notification: meaningless.  */ +# define SIGEV_NONE	SIGEV_NONE +  SIGEV_THREAD,			/* Deliver via thread creation.  */ +# define SIGEV_THREAD	SIGEV_THREAD + +  SIGEV_THREAD_ID = 4		/* Send signal to specific thread.  */ +#define SIGEV_THREAD_ID	SIGEV_THREAD_ID +}; + +#endif	/* have _SIGNAL_H.  */ diff --git a/libc/sysdeps/linux/sparc64/bits/signalfd.h b/libc/sysdeps/linux/sparc64/bits/signalfd.h new file mode 100644 index 000000000..881547c78 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/signalfd.h @@ -0,0 +1,28 @@ +/* Copyright (C) 2007-2017 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef	_SYS_SIGNALFD_H +# error "Never use <bits/signalfd.h> directly; include <sys/signalfd.h> instead." +#endif + +/* Flags for signalfd.  */ +enum +  { +    SFD_CLOEXEC = 0x400000, +#define SFD_CLOEXEC SFD_CLOEXEC +    SFD_NONBLOCK = 0x004000 +#define SFD_NONBLOCK SFD_NONBLOCK +  }; diff --git a/libc/sysdeps/linux/sparc64/bits/signum.h b/libc/sysdeps/linux/sparc64/bits/signum.h new file mode 100644 index 000000000..b63bf39c1 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/signum.h @@ -0,0 +1,62 @@ +/* Signal number definitions.  Linux/SPARC version. +   Copyright (C) 1996, 1997, 1998, 2003 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifdef	_SIGNAL_H + +/* + * Linux/SPARC has different signal numbers that Linux/i386: I'm trying + * to make it OSF/1 binary compatible, at least for normal binaries. + */ +#define SIGHUP		 1 +#define SIGINT		 2 +#define SIGQUIT		 3 +#define SIGILL		 4 +#define SIGTRAP		 5 +#define SIGABRT		 6 +#define SIGIOT		 6 +#define SIGEMT           7 +#define SIGFPE		 8 +#define SIGKILL		 9 +#define SIGBUS          10 +#define SIGSEGV		11 +#define SIGSYS		12 +#define SIGPIPE		13 +#define SIGALRM		14 +#define SIGTERM		15 +#define SIGURG          16 + +/* SunOS values which deviate from the Linux/i386 ones */ +#define SIGSTOP		17 +#define SIGTSTP		18 +#define SIGCONT		19 +#define SIGCHLD		20 +#define SIGCLD		SIGCHLD	/* Same as SIGCHLD (System V).  */ +#define SIGTTIN		21 +#define SIGTTOU		22 +#define SIGIO		23 +#define SIGPOLL		SIGIO   /* SysV name for SIGIO */ +#define SIGXCPU		24 +#define SIGXFSZ		25 +#define SIGVTALRM	26 +#define SIGPROF		27 +#define SIGWINCH	28 +#define SIGLOST		29 +#define SIGPWR          SIGLOST +#define SIGUSR1		30 +#define SIGUSR2		31 + +#endif	/* <signal.h> included.  */ diff --git a/libc/sysdeps/linux/sparc64/bits/sigstack.h b/libc/sysdeps/linux/sparc64/bits/sigstack.h new file mode 100644 index 000000000..58533fd61 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/sigstack.h @@ -0,0 +1,55 @@ +/* sigstack, sigaltstack definitions. +   Copyright (C) 1998, 1999, 2002 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SIGNAL_H +# error "Never include this file directly.  Use <signal.h> instead" +#endif + + +#if defined __UCLIBC_SUSV4_LEGACY__ || !defined __STRICT_HEADERS__ +/* Structure describing a signal stack (obsolete).  */ +struct sigstack +  { +    void *ss_sp;		/* Signal stack pointer.  */ +    int ss_onstack;		/* Nonzero if executing on this stack.  */ +  }; +#endif + + +/* Possible values for `ss_flags.'.  */ +enum +{ +  SS_ONSTACK = 1, +#define SS_ONSTACK	SS_ONSTACK +  SS_DISABLE +#define SS_DISABLE	SS_DISABLE +}; + +/* Minimum stack size for a signal handler.  */ +#define MINSIGSTKSZ	4096 + +/* System default stack size.  */ +#define SIGSTKSZ	16384 + + +/* Alternate, preferred interface.  */ +typedef struct sigaltstack +  { +    void *ss_sp; +    int ss_flags; +    size_t ss_size; +  } stack_t; diff --git a/libc/sysdeps/linux/sparc64/bits/socket_type.h b/libc/sysdeps/linux/sparc64/bits/socket_type.h new file mode 100644 index 000000000..aff7fa132 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/socket_type.h @@ -0,0 +1,53 @@ +/* System-specific socket constants and types.  Linux version. +   Copyright (C) 1991,1992,1994-2001,2004,2006 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_SOCKET_H +# error "Never include <bits/socket_type.h> directly; use <sys/socket.h> instead." +#endif + +/* Types of sockets.  */ +enum __socket_type +{ +  SOCK_STREAM = 1,		/* Sequenced, reliable, connection-based +				   byte streams.  */ +#define SOCK_STREAM SOCK_STREAM +  SOCK_DGRAM = 2,		/* Connectionless, unreliable datagrams +				   of fixed maximum length.  */ +#define SOCK_DGRAM SOCK_DGRAM +  SOCK_RAW = 3,			/* Raw protocol interface.  */ +#define SOCK_RAW SOCK_RAW +  SOCK_RDM = 4,			/* Reliably-delivered messages.  */ +#define SOCK_RDM SOCK_RDM +  SOCK_SEQPACKET = 5,		/* Sequenced, reliable, connection-based, +				   datagrams of fixed maximum length.  */ +#define SOCK_SEQPACKET SOCK_SEQPACKET +  SOCK_DCCP = 6,		/* Datagram Congestion Control Protocol.  */ +#define SOCK_DCCP SOCK_DCCP +  SOCK_PACKET = 10,		/* Linux specific way of getting packets +				   at the dev level.  For writing rarp and +				   other similar things on the user level. */ +#define SOCK_PACKET SOCK_PACKET + +  /* Flags to be ORed into the type parameter of socket and socketpair.  */ + +  SOCK_CLOEXEC = 0x400000,	/* Atomically set close-on-exec flag for the +				   new descriptor(s).  */ +#define SOCK_CLOEXEC SOCK_CLOEXEC +  SOCK_NONBLOCK = 0x004000	/* Atomically mark descriptor(s) as +				   non-blocking.  */ +#define SOCK_NONBLOCK SOCK_NONBLOCK +}; diff --git a/libc/sysdeps/linux/sparc64/bits/stackinfo.h b/libc/sysdeps/linux/sparc64/bits/stackinfo.h new file mode 100644 index 000000000..e402320e7 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/stackinfo.h @@ -0,0 +1,26 @@ +/* Copyright (C) 2001 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +/* This file contains a bit of information about the stack allocation +   of the processor.  */ + +#ifndef _STACKINFO_H +#define _STACKINFO_H	1 + +/* On sparc the stack grows down.  */ +#define _STACK_GROWS_DOWN	1 + +#endif	/* stackinfo.h */ diff --git a/libc/sysdeps/linux/sparc64/bits/stat.h b/libc/sysdeps/linux/sparc64/bits/stat.h new file mode 100644 index 000000000..cae973b4e --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/stat.h @@ -0,0 +1,167 @@ +/* Copyright (C) 1992, 1995-2002, 2006, 2009 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_STAT_H +# error "Never include <bits/stat.h> directly; use <sys/stat.h> instead." +#endif + +/* Versions of the `struct stat' data structure.  */ +#define _STAT_VER_LINUX_OLD	1 +#define _STAT_VER_KERNEL	1 +#define _STAT_VER_SVR4		2 +#define _STAT_VER_LINUX		3 +#define _STAT_VER		_STAT_VER_LINUX	/* The one defined below.  */ + +/* Versions of the `xmknod' interface.  */ +#define _MKNOD_VER_LINUX	1 +#define _MKNOD_VER_SVR4		2 +#define _MKNOD_VER		_MKNOD_VER_LINUX /* The bits defined below.  */ + + +struct stat +  { +    __dev_t st_dev;			/* Device.  */ +#if __WORDSIZE == 64 || !defined __USE_FILE_OFFSET64 +    unsigned short int __pad1; +    __ino_t st_ino;			/* File serial number.	*/ +#else +    __ino64_t st_ino;			/* File serial number.	*/ +#endif +    __mode_t st_mode;			/* File mode.  */ +    __nlink_t st_nlink;			/* Link count.  */ +    __uid_t st_uid;			/* User ID of the file's owner.	*/ +    __gid_t st_gid;			/* Group ID of the file's group.*/ +    __dev_t st_rdev;			/* Device number, if device.  */ +    unsigned short int __pad2; +#ifndef __USE_FILE_OFFSET64 +    __off_t st_size;			/* Size of file, in bytes.  */ +#else +    __off64_t st_size;			/* Size of file, in bytes.  */ +#endif +    __blksize_t st_blksize;		/* Optimal block size for I/O.  */ + +#ifndef __USE_FILE_OFFSET64 +    __blkcnt_t st_blocks;		/* Number 512-byte blocks allocated. */ +#else +    __blkcnt64_t st_blocks;		/* Number 512-byte blocks allocated. */ +#endif +#ifdef __USE_MISC +    /* Nanosecond resolution timestamps are stored in a format +       equivalent to 'struct timespec'.  This is the type used +       whenever possible but the Unix namespace rules do not allow the +       identifier 'timespec' to appear in the <sys/stat.h> header. +       Therefore we have to handle the use of this header in strictly +       standard-compliant sources special.  */ +    struct timespec st_atim;		/* Time of last access.  */ +    struct timespec st_mtim;		/* Time of last modification.  */ +    struct timespec st_ctim;		/* Time of last status change.  */ +# define st_atime st_atim.tv_sec	/* Backward compatibility.  */ +# define st_mtime st_mtim.tv_sec +# define st_ctime st_ctim.tv_sec +#else +    __time_t st_atime;			/* Time of last access.  */ +    unsigned long int st_atimensec;	/* Nscecs of last access.  */ +    __time_t st_mtime;			/* Time of last modification.  */ +    unsigned long int st_mtimensec;	/* Nsecs of last modification.  */ +    __time_t st_ctime;			/* Time of last status change.  */ +    unsigned long int st_ctimensec;	/* Nsecs of last status change.  */ +#endif +    unsigned long int __unused4; +    unsigned long int __unused5; +  }; + +#ifdef __USE_LARGEFILE64 +struct stat64 +  { +    __dev_t st_dev;			/* Device.  */ +#if __WORDSIZE == 64 +    unsigned short int __pad1; +#endif +    __ino64_t st_ino;			/* File serial number.	*/ +    __mode_t st_mode;			/* File mode.  */ +    __nlink_t st_nlink;			/* Link count.  */ +    __uid_t st_uid;			/* User ID of the file's owner.	*/ +    __gid_t st_gid;			/* Group ID of the file's group.*/ +    __dev_t st_rdev;			/* Device number, if device.  */ +    unsigned short int __pad2; +    __off64_t st_size;			/* Size of file, in bytes.  */ +    __blksize_t st_blksize;		/* Optimal block size for I/O.  */ + +    __blkcnt64_t st_blocks;		/* Number 512-byte blocks allocated. */ +#ifdef __USE_MISC +    /* Nanosecond resolution timestamps are stored in a format +       equivalent to 'struct timespec'.  This is the type used +       whenever possible but the Unix namespace rules do not allow the +       identifier 'timespec' to appear in the <sys/stat.h> header. +       Therefore we have to handle the use of this header in strictly +       standard-compliant sources special.  */ +    struct timespec st_atim;		/* Time of last access.  */ +    struct timespec st_mtim;		/* Time of last modification.  */ +    struct timespec st_ctim;		/* Time of last status change.  */ +# define st_atime st_atim.tv_sec	/* Backward compatibility.  */ +# define st_mtime st_mtim.tv_sec +# define st_ctime st_ctim.tv_sec +#else +    __time_t st_atime;			/* Time of last access.  */ +    unsigned long int st_atimensec;	/* Nscecs of last access.  */ +    __time_t st_mtime;			/* Time of last modification.  */ +    unsigned long int st_mtimensec;	/* Nsecs of last modification.  */ +    __time_t st_ctime;			/* Time of last status change.  */ +    unsigned long int st_ctimensec;	/* Nsecs of last status change.  */ +#endif +    unsigned long int __unused4; +    unsigned long int __unused5; +  }; +#endif + +/* Tell code we have these members.  */ +#define	_STATBUF_ST_BLKSIZE +#define _STATBUF_ST_RDEV +/* Nanosecond resolution time values are supported.  */ +#define _STATBUF_ST_NSEC + +/* Encoding of the file mode.  */ + +#define	__S_IFMT	0170000	/* These bits determine file type.  */ + +/* File types.  */ +#define	__S_IFDIR	0040000	/* Directory.  */ +#define	__S_IFCHR	0020000	/* Character device.  */ +#define	__S_IFBLK	0060000	/* Block device.  */ +#define	__S_IFREG	0100000	/* Regular file.  */ +#define	__S_IFIFO	0010000	/* FIFO.  */ +#define	__S_IFLNK	0120000	/* Symbolic link.  */ +#define	__S_IFSOCK	0140000	/* Socket.  */ + +/* POSIX.1b objects.  Note that these macros always evaluate to zero.  But +   they do it by enforcing the correct use of the macros.  */ +#define __S_TYPEISMQ(buf)  ((buf)->st_mode - (buf)->st_mode) +#define __S_TYPEISSEM(buf) ((buf)->st_mode - (buf)->st_mode) +#define __S_TYPEISSHM(buf) ((buf)->st_mode - (buf)->st_mode) + +/* Protection bits.  */ + +#define	__S_ISUID	04000	/* Set user ID on execution.  */ +#define	__S_ISGID	02000	/* Set group ID on execution.  */ +#define	__S_ISVTX	01000	/* Save swapped text after use (sticky).  */ +#define	__S_IREAD	0400	/* Read by owner.  */ +#define	__S_IWRITE	0200	/* Write by owner.  */ +#define	__S_IEXEC	0100	/* Execute by owner.  */ + +#ifdef __USE_ATFILE +# define UTIME_NOW	((1l << 30) - 1l) +# define UTIME_OMIT	((1l << 30) - 2l) +#endif diff --git a/libc/sysdeps/linux/sparc64/bits/syscalls.h b/libc/sysdeps/linux/sparc64/bits/syscalls.h new file mode 100644 index 000000000..bf3f7a0f2 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/syscalls.h @@ -0,0 +1,116 @@ +/* + * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball. + * ported from GNU C Library + */ + +/* Copyright (C) 2000-2017 Free Software Foundation, Inc. + +The GNU C Library is free software; you can redistribute it and/or +modify it under the terms of the GNU Lesser General Public License as +published by the Free Software Foundation; either version 2.1 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 +Lesser General Public License for more details. + +You should have received a copy of the GNU Lesser General Public +License along with the GNU C Library; if not, see +<http://www.gnu.org/licenses/>.  */ + +#ifndef _BITS_SYSCALLS_H +#define _BITS_SYSCALLS_H +#ifndef _SYSCALL_H +# error "Never use <bits/syscalls.h> directly; include <sys/syscall.h> instead." +#endif + +#ifndef __ASSEMBLER__ + +#define __SYSCALL_CLOBBERS						\ +	"f0", "f1", "f2", "f3", "f4", "f5", "f6", "f7",			\ +	"f8", "f9", "f10", "f11", "f12", "f13", "f14", "f15",		\ +	"f16", "f17", "f18", "f19", "f20", "f21", "f22", "f23",		\ +	"f24", "f25", "f26", "f27", "f28", "f29", "f30", "f31",		\ +	"f32", "f34", "f36", "f38", "f40", "f42", "f44", "f46",		\ +	"f48", "f50", "f52", "f54", "f56", "f58", "f60", "f62",		\ +	"cc", "memory" + +#define __SYSCALL_STRING						\ +	"ta	0x6d;"							\ +	"bcc,pt	%%xcc, 1f;"						\ +	" mov	0, %%g1;"						\ +	"sub	%%g0, %%o0, %%o0;"					\ +	"mov	1, %%g1;"						\ +	"1:" + +#define INTERNAL_SYSCALL_NCS(name, err, nr, args...) 		\ +({                                               		\ +            register long __o0 __asm__("o0");            	\ +	    long err = name;					\ +            LOAD_ARGS_##nr(args)                           	\ +								\ +            __asm__ __volatile__( __SYSCALL_STRING          	\ +                : "=r" (err), "=r" (__o0)                   	\ +                : "0" (err), "1" (__o0) ASM_ARGS_##nr       	\ +                : __SYSCALL_CLOBBERS );	                    	\ +            __o0;					    	\ +}) + +#define INTERNAL_SYSCALL_ERROR_P(val, err) \ +  ((unsigned int) (val) >= 0xfffff001u) + +#define INLINE_CLONE_SYSCALL(arg1,arg2,arg3,arg4,arg5)			\ +({									\ +	register long __o0 __asm__ ("o0") = (long)(arg1);		\ +	register long __o1 __asm__ ("o1") = (long)(arg2);		\ +	register long __o2 __asm__ ("o2") = (long)(arg3);		\ +	register long __o3 __asm__ ("o3") = (long)(arg4);		\ +	register long __o4 __asm__ ("o4") = (long)(arg5);		\ +	register long __g1 __asm__ ("g1") = __NR_clone;			\ +	__asm __volatile (__SYSCALL_STRING :				\ +			  "=r" (__g1), "=r" (__o0), "=r" (__o1)	:	\ +			  "0" (__g1), "1" (__o0), "2" (__o1),		\ +			  "r" (__o2), "r" (__o3), "r" (__o4) :		\ +			  __SYSCALL_CLOBBERS);				\ +	if (INTERNAL_SYSCALL_ERROR_P (__o0, __g1))			\ +	  {		     			       		   	\ +	    __set_errno (INTERNAL_SYSCALL_ERRNO (__o0, __g1));		\ +	    __o0 = -1L;			    				\ +	  } 	      							\ +	else								\ +	  { 	      							\ +	    __o0 &= (__o1 - 1);						\ +	  } 	    	    						\ +	__o0;								\ +}) + +#define LOAD_ARGS_0() +#define ASM_ARGS_0 +#define LOAD_ARGS_1(o0) \ +    __o0 = (long)o0;     \ +    LOAD_ARGS_0() +#define ASM_ARGS_1 ASM_ARGS_0, "r" (__o0) +#define LOAD_ARGS_2(o0, o1)			\ +  register long __o1 __asm__ ("o1") = (long) (o1);	\ +  LOAD_ARGS_1 (o0) +#define ASM_ARGS_2	ASM_ARGS_1, "r" (__o1) +#define LOAD_ARGS_3(o0, o1, o2)			\ +  register long __o2 __asm__ ("o2") = (long) (o2);	\ +  LOAD_ARGS_2 (o0, o1) +#define ASM_ARGS_3	ASM_ARGS_2, "r" (__o2) +#define LOAD_ARGS_4(o0, o1, o2, o3)		\ +  register long __o3 __asm__ ("o3") = (long) (o3);	\ +  LOAD_ARGS_3 (o0, o1, o2) +#define ASM_ARGS_4	ASM_ARGS_3, "r" (__o3) +#define LOAD_ARGS_5(o0, o1, o2, o3, o4)		\ +  register long __o4 __asm__ ("o4") = (long) (o4);	\ +  LOAD_ARGS_4 (o0, o1, o2, o3) +#define ASM_ARGS_5	ASM_ARGS_4, "r" (__o4) +#define LOAD_ARGS_6(o0, o1, o2, o3, o4, o5)	\ +  register long __o5 __asm__ ("o5") = (long) (o5);	\ +  LOAD_ARGS_5 (o0, o1, o2, o3, o4) +#define ASM_ARGS_6	ASM_ARGS_5, "r" (__o5) + +#endif /* __ASSEMBLER__ */ +#endif /* _BITS_SYSCALLS_H */ diff --git a/libc/sysdeps/linux/sparc64/bits/termios.h b/libc/sysdeps/linux/sparc64/bits/termios.h new file mode 100644 index 000000000..f4800d1b1 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/termios.h @@ -0,0 +1,231 @@ +/* termios type and macro definitions.  Linux/SPARC version. +   Copyright (C) 1993, 1994, 1995, 1996, 1997, 2000, 2005 +       Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _TERMIOS_H +# error "Never include <bits/termios.h> directly; use <termios.h> instead." +#endif + +typedef unsigned char cc_t; +typedef unsigned int speed_t; +typedef unsigned int tcflag_t; + +#define NCCS 17 +struct termios +  { +    tcflag_t c_iflag;		/* input mode flags */ +    tcflag_t c_oflag;		/* output mode flags */ +    tcflag_t c_cflag;		/* control mode flags */ +    tcflag_t c_lflag;		/* local mode flags */ +    cc_t c_line;		/* line discipline */ +    cc_t c_cc[NCCS];		/* control characters */ +  }; + +/* c_cc characters */ +#define VINTR    0 +#define VQUIT    1 +#define VERASE   2 +#define VKILL    3 +#define VEOF     4 +#define VEOL     5 +#define VEOL2    6 +#define VSWTC    7 +#define VSTART   8 +#define VSTOP    9 +#define VSUSP    10 +#define VDSUSP   11		/* SunOS POSIX nicety I do believe... */ +#define VREPRINT 12 +#define VDISCARD 13 +#define VWERASE  14 +#define VLNEXT   15 + +/* User apps assume vmin/vtime is shared with eof/eol */ +#define VMIN     VEOF +#define VTIME    VEOL + +/* c_iflag bits */ +#define IGNBRK	0x00000001 +#define BRKINT	0x00000002 +#define IGNPAR	0x00000004 +#define PARMRK	0x00000008 +#define INPCK	0x00000010 +#define ISTRIP	0x00000020 +#define INLCR	0x00000040 +#define IGNCR	0x00000080 +#define ICRNL	0x00000100 +#define IUCLC	0x00000200 +#define IXON	0x00000400 +#define IXANY	0x00000800 +#define IXOFF	0x00001000 +#define IMAXBEL	0x00002000 +#define IUTF8	0x00004000 + +/* c_oflag bits */ +#define OPOST	0x00000001 +#define OLCUC	0x00000002 +#define ONLCR	0x00000004 +#define OCRNL	0x00000008 +#define ONOCR	0x00000010 +#define ONLRET	0x00000020 +#define OFILL	0x00000040 +#define OFDEL	0x00000080 +#if defined __USE_MISC || defined __USE_XOPEN +# define NLDLY	0x00000100 +# define   NL0	0x00000000 +# define   NL1	0x00000100 +# define CRDLY	0x00000600 +# define   CR0	0x00000000 +# define   CR1	0x00000200 +# define   CR2	0x00000400 +# define   CR3	0x00000600 +# define TABDLY	0x00001800 +# define   TAB0	0x00000000 +# define   TAB1	0x00000800 +# define   TAB2	0x00001000 +# define   TAB3	0x00001800 +# define BSDLY	0x00002000 +# define   BS0	0x00000000 +# define   BS1	0x00002000 +#define FFDLY	0x00008000 +#define   FF0	0x00000000 +#define   FF1	0x00008000 +#endif +#define VTDLY	0x00004000 +#define   VT0	0x00000000 +#define   VT1	0x00004000 +#define PAGEOUT 0x00010000	/* SUNOS specific */ +#define WRAP    0x00020000	/* SUNOS specific */ + +#ifdef __USE_MISC +# define   XTABS	0x00001800 +#endif + +/* c_cflag bit meaning */ +#ifdef __USE_MISC +# define CBAUD	0x0000100f +#endif +#define  B0	0x00000000	/* hang up */ +#define  B50	0x00000001 +#define  B75	0x00000002 +#define  B110	0x00000003 +#define  B134	0x00000004 +#define  B150	0x00000005 +#define  B200	0x00000006 +#define  B300	0x00000007 +#define  B600	0x00000008 +#define  B1200	0x00000009 +#define  B1800	0x0000000a +#define  B2400	0x0000000b +#define  B4800	0x0000000c +#define  B9600	0x0000000d +#define  B19200	0x0000000e +#define  B38400	0x0000000f +#ifdef __USE_MISC +# define EXTA    B19200 +# define EXTB    B38400 +#endif +#define  CSIZE  0x00000030 +#define   CS5	0x00000000 +#define   CS6	0x00000010 +#define   CS7	0x00000020 +#define   CS8	0x00000030 +#define CSTOPB	0x00000040 +#define CREAD	0x00000080 +#define PARENB	0x00000100 +#define PARODD	0x00000200 +#define HUPCL	0x00000400 +#define CLOCAL	0x00000800 +#ifdef __USE_MISC +# define CBAUDEX 0x00001000 +#endif +#define  B57600  0x00001001 +#define  B115200 0x00001002 +#define  B230400 0x00001003 +#define  B460800 0x00001004 +#define  B76800  0x00001005 +#define  B153600 0x00001006 +#define  B307200 0x00001007 +#define  B614400 0x00001008 +#define  B921600 0x00001009 +#define  B500000 0x0000100a +#define  B576000 0x0000100b +#define B1000000 0x0000100c +#define B1152000 0x0000100d +#define B1500000 0x0000100e +#define B2000000 0x0000100f +#define __MAX_BAUD B2000000 + +#ifdef __USE_MISC +# define CIBAUD	 0x100f0000	/* input baud rate (not used) */ +# define CMSPAR	 0x40000000	/* mark or space (stick) parity */ +# define CRTSCTS 0x80000000	/* flow control */ +#endif + +/* c_lflag bits */ +#define ISIG	0x00000001 +#define ICANON	0x00000002 +#if defined __USE_MISC || defined __USE_XOPEN +# define XCASE	0x00000004 +#endif +#define ECHO	0x00000008 +#define ECHOE	0x00000010 +#define ECHOK	0x00000020 +#define ECHONL	0x00000040 +#define NOFLSH	0x00000080 +#define TOSTOP	0x00000100 +#ifdef __USE_MISC +# define ECHOCTL	0x00000200 +# define ECHOPRT	0x00000400 +# define ECHOKE		0x00000800 +# define DEFECHO 	0x00001000	/* SUNOS thing, what is it? */ +# define FLUSHO		0x00002000 +# define PENDIN		0x00004000 +#endif +#define IEXTEN	0x00008000 + +/* modem lines */ +#define TIOCM_LE	0x001 +#define TIOCM_DTR	0x002 +#define TIOCM_RTS	0x004 +#define TIOCM_ST	0x008 +#define TIOCM_SR	0x010 +#define TIOCM_CTS	0x020 +#define TIOCM_CAR	0x040 +#define TIOCM_RNG	0x080 +#define TIOCM_DSR	0x100 +#define TIOCM_CD	TIOCM_CAR +#define TIOCM_RI	TIOCM_RNG + +/* ioctl (fd, TIOCSERGETLSR, &result) where result may be as below */ +#define TIOCSER_TEMT    0x01	/* Transmitter physically empty */ + + +/* tcflow() and TCXONC use these */ +#define	TCOOFF		0 +#define	TCOON		1 +#define	TCIOFF		2 +#define	TCION		3 + +/* tcflush() and TCFLSH use these */ +#define	TCIFLUSH	0 +#define	TCOFLUSH	1 +#define	TCIOFLUSH	2 + +/* tcsetattr uses these */ +#define	TCSANOW		0 +#define	TCSADRAIN	1 +#define	TCSAFLUSH	2 diff --git a/libc/sysdeps/linux/sparc64/bits/timerfd.h b/libc/sysdeps/linux/sparc64/bits/timerfd.h new file mode 100644 index 000000000..1460866f9 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/timerfd.h @@ -0,0 +1,28 @@ +/* Copyright (C) 2008-2012 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef	_SYS_TIMERFD_H +# error "Never use <bits/timerfd.h> directly; include <sys/timerfd.h> instead." +#endif + +/* Bits to be set in the FLAGS parameter of `timerfd_create'.  */ +enum +  { +    TFD_CLOEXEC = 0x400000, +#define TFD_CLOEXEC TFD_CLOEXEC +    TFD_NONBLOCK = 0x004000 +#define TFD_NONBLOCK TFD_NONBLOCK +  }; diff --git a/libc/sysdeps/linux/sparc64/bits/typesizes.h b/libc/sysdeps/linux/sparc64/bits/typesizes.h new file mode 100644 index 000000000..eba385661 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/typesizes.h @@ -0,0 +1,64 @@ +/* bits/typesizes.h -- underlying types for *_t.  Linux/SPARC version. +   Copyright (C) 2002, 2003 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _BITS_TYPES_H +# error "Never include <bits/typesizes.h> directly; use <sys/types.h> instead." +#endif + +#ifndef	_BITS_TYPESIZES_H +#define	_BITS_TYPESIZES_H	1 + +/* See <bits/types.h> for the meaning of these macros.  This file exists so +   that <bits/types.h> need not vary across different GNU platforms.  */ + +#define __DEV_T_TYPE		__UQUAD_TYPE +#define __UID_T_TYPE		__U32_TYPE +#define __GID_T_TYPE		__U32_TYPE +#define __INO_T_TYPE		__ULONGWORD_TYPE +#define __INO64_T_TYPE		__UQUAD_TYPE +#define __MODE_T_TYPE		__U32_TYPE +#define __NLINK_T_TYPE		__U32_TYPE +#define __OFF_T_TYPE		__SLONGWORD_TYPE +#define __OFF64_T_TYPE		__SQUAD_TYPE +#define __PID_T_TYPE		__S32_TYPE +#define __RLIM_T_TYPE		__ULONGWORD_TYPE +#define __RLIM64_T_TYPE		__UQUAD_TYPE +#define	__BLKCNT_T_TYPE		__SLONGWORD_TYPE +#define	__BLKCNT64_T_TYPE	__SQUAD_TYPE +#define	__FSBLKCNT_T_TYPE	__ULONGWORD_TYPE +#define	__FSBLKCNT64_T_TYPE	__UQUAD_TYPE +#define	__FSFILCNT_T_TYPE	__ULONGWORD_TYPE +#define	__FSFILCNT64_T_TYPE	__UQUAD_TYPE +#define	__ID_T_TYPE		__U32_TYPE +#define __CLOCK_T_TYPE		__SLONGWORD_TYPE +#define __TIME_T_TYPE		__SLONGWORD_TYPE +#define __USECONDS_T_TYPE	__U32_TYPE +#define __SUSECONDS_T_TYPE	__S32_TYPE +#define __DADDR_T_TYPE		__S32_TYPE +#define __SWBLK_T_TYPE		__SLONGWORD_TYPE +#define __KEY_T_TYPE		__S32_TYPE +#define __CLOCKID_T_TYPE	__S32_TYPE +#define __TIMER_T_TYPE		void * +#define __BLKSIZE_T_TYPE	__SLONGWORD_TYPE +#define __FSID_T_TYPE		struct { int __val[2]; } +#define __SSIZE_T_TYPE		__SWORD_TYPE + +/* Number of descriptors that can fit in an `fd_set'.  */ +#define	__FD_SETSIZE		1024 + + +#endif /* bits/typesizes.h */ diff --git a/libc/sysdeps/linux/sparc64/bits/uClibc_arch_features.h b/libc/sysdeps/linux/sparc64/bits/uClibc_arch_features.h new file mode 100644 index 000000000..283a250bb --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/uClibc_arch_features.h @@ -0,0 +1,41 @@ +/* + * Track misc arch-specific features that aren't config options + */ + +#ifndef _BITS_UCLIBC_ARCH_FEATURES_H +#define _BITS_UCLIBC_ARCH_FEATURES_H + +/* instruction used when calling abort() to kill yourself */ +#define __UCLIBC_ABORT_INSTRUCTION__ "unimp 0xf00" + +/* can your target use syscall6() for mmap ? */ +#define __UCLIBC_MMAP_HAS_6_ARGS__ + +/* does your target align 64bit values in register pairs ? (32bit arches only) */ +#undef __UCLIBC_SYSCALL_ALIGN_64BIT__ + +/* does your target have a broken create_module() ? */ +#undef __UCLIBC_BROKEN_CREATE_MODULE__ + +/* does your target have to worry about older [gs]etrlimit() ? */ +#undef __UCLIBC_HANDLE_OLDER_RLIMIT__ + +/* does your target have an asm .set ? */ +#define __UCLIBC_HAVE_ASM_SET_DIRECTIVE__ + +/* define if target supports .weak */ +#define __UCLIBC_HAVE_ASM_WEAK_DIRECTIVE__ + +/* define if target supports .weakext */ +#undef __UCLIBC_HAVE_ASM_WEAKEXT_DIRECTIVE__ + +/* define if target supports CFI pseudo ops */ +#define __UCLIBC_HAVE_ASM_CFI_DIRECTIVES__ + +/* define if target supports IEEE signed zero floats */ +#define __UCLIBC_HAVE_SIGNED_ZERO__ + +/* only weird assemblers generally need this */ +#undef __UCLIBC_ASM_LINE_SEP__ + +#endif /* _BITS_UCLIBC_ARCH_FEATURES_H */ diff --git a/libc/sysdeps/linux/sparc64/bits/uClibc_page.h b/libc/sysdeps/linux/sparc64/bits/uClibc_page.h new file mode 100644 index 000000000..54da57a0f --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/uClibc_page.h @@ -0,0 +1,17 @@ +/* + * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball. + */ + +/* Supply an architecture specific value for PAGE_SIZE and friends.  */ + +#ifndef _UCLIBC_PAGE_H +#define _UCLIBC_PAGE_H + +/* PAGE_SHIFT determines the page size -- in this case 8192 */ +#define PAGE_SHIFT	13 +#define PAGE_SIZE	(1UL << PAGE_SHIFT) +#define PAGE_MASK	(~(PAGE_SIZE-1)) + +#define MMAP2_PAGE_SHIFT PAGE_SHIFT + +#endif /* _UCLIBC_PAGE_H */ diff --git a/libc/sysdeps/linux/sparc64/bits/wordsize.h b/libc/sysdeps/linux/sparc64/bits/wordsize.h new file mode 100644 index 000000000..786a00087 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bits/wordsize.h @@ -0,0 +1,4 @@ +/* + * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball. + */ +#define __WORDSIZE	64 diff --git a/libc/sysdeps/linux/sparc64/brk.S b/libc/sysdeps/linux/sparc64/brk.S new file mode 100644 index 000000000..2d857af6c --- /dev/null +++ b/libc/sysdeps/linux/sparc64/brk.S @@ -0,0 +1,101 @@ +/* Copyright (C) 1997-2017 Free Software Foundation, Inc. +   Contributed by Richard Henderson <richard@gnu.ai.mit.edu>, 1997. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +/* __brk is a special syscall under Linux since it never returns an +   error.  Instead, the error condition is indicated by returning the old +   break value (instead of the new, requested one).  */ + +#include <sysdep.h> +#define _ERRNO_H +#include <bits/errno.h> + +#ifdef __PIC__ +.section .bss +	.align 8 +	.globl __curbrk +__curbrk: .skip 8 +	.type __curbrk,@object +	.size __curbrk,8 +#else +.common __curbrk, 8, 8 +#endif + +	.text +ENTRY (__brk) +	save	%sp, -192, %sp +	cfi_def_cfa_register(%fp) +	cfi_window_save +	cfi_register(%o7, %i7) +#ifdef __PIC__ +	SETUP_PIC_REG(l7) +#endif + +	LOADSYSCALL(brk) +	mov	%i0, %o0 + +	ta	0x6d + +	/* All the ways we can fail... */ +	bcs,pn	%xcc, .Lerr1 +	 nop +	brz,pt	%i0, .Lok +	 subcc	%i0, %o0, %g0 +	bne,pn	%xcc, .Lerr0 +	 nop + +	/* Update __curbrk and return cleanly.  */ +.Lok: +#ifndef __PIC__ +	sethi	%hi(__curbrk), %g1 +	or	%g1, %lo(__curbrk), %g1 +#else +	sethi	%gdop_hix22(__curbrk), %g1 +	xor	%g1, %gdop_lox10(__curbrk), %g1 +	ldx	[%l7 + %g1], %g1, %gdop(__curbrk) +#endif +	stx	%o0, [%g1] +	mov	%g0, %i0 + +	/* Don't use "ret" cause the preprocessor will eat it.  */ +	jmpl	%i7+8, %g0 +	 restore + +	/* What a horrible way to die.  */ +.Lerr0:	set	ENOMEM, %o0 +.Lerr1: +#ifndef _LIBC_REENTRANT +#ifndef __PIC__ +	sethi	%hi(errno), %g1 +	or	%g1, %lo(errno), %g1 +#else +	sethi	%gdop_hix22(errno), %g1 +	xor	%g1, %gdop_lox10(errno), %g1 +	ldx	[%l7 + %g1], %g1, %gdop(errno) +#endif +	st	%o0, [%g1] +#else +	call	HIDDEN_JUMPTARGET(__errno_location) +	 mov	%o0,%l1 +	st	%l1, [%o0] +#endif +	sub	%g0, 1, %i0 +	jmpl	%i7+8, %g0 +	 restore +END (__brk) + +weak_alias (__brk, brk) +libc_hidden_def(brk) diff --git a/libc/sysdeps/linux/sparc64/bsd-_setjmp.S b/libc/sysdeps/linux/sparc64/bsd-_setjmp.S new file mode 100644 index 000000000..4e6a2da56 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bsd-_setjmp.S @@ -0,0 +1 @@ +/* _setjmp is in setjmp.S  */ diff --git a/libc/sysdeps/linux/sparc64/bsd-setjmp.S b/libc/sysdeps/linux/sparc64/bsd-setjmp.S new file mode 100644 index 000000000..1da848d2f --- /dev/null +++ b/libc/sysdeps/linux/sparc64/bsd-setjmp.S @@ -0,0 +1 @@ +/* setjmp is in setjmp.S  */ diff --git a/libc/sysdeps/linux/sparc64/clone.S b/libc/sysdeps/linux/sparc64/clone.S new file mode 100644 index 000000000..63ab38bb7 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/clone.S @@ -0,0 +1,85 @@ +/* Copyright (C) 1997-2017 Free Software Foundation, Inc. +   Contributed by Richard Henderson (rth@tamu.edu). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +/* clone() is even more special than fork() as it mucks with stacks +   and invokes a function in the right context after its all over.  */ + +#include <asm/errno.h> +#include <asm/unistd.h> +#include <sysdep.h> + +#define CLONE_VM	0x00000100 + +/* int clone(int (*fn)(void *arg), void *child_stack, int flags, void *arg, +	     pid_t *ptid, void *tls, pid_t *ctid); */ + +	.register	%g2,#scratch +	.register	%g3,#scratch + +	.text + +ENTRY (__clone) +	save	%sp, -192, %sp +	cfi_def_cfa_register(%fp) +	cfi_window_save +	cfi_register(%o7, %i7) + +	/* sanity check arguments */ +	brz,pn	%i0, 99f		/* fn non-NULL? */ +	 mov	%i0, %g2 +	brz,pn	%i1, 99f		/* child_stack non-NULL? */ +	 mov	%i2, %o0		/* clone flags */ + +	/* The child_stack is the top of the stack, allocate one +	   whole stack frame from that as this is what the kernel +	   expects.  Also, subtract STACK_BIAS.  */ +	sub	%i1, 192 + 0x7ff, %o1 +	mov	%i3, %g3 + +	mov	%i4,%o2			/* PTID */ +	mov	%i5,%o3			/* TLS */ +	ldx	[%fp+0x7ff+176],%o4	/* CTID */ + +	/* Do the system call */ +	set	__NR_clone, %g1 +	ta	0x6d +	bcs,pn	%xcc, 98f +	 nop +	brnz,pn	%o1, __thread_start +	 nop +	jmpl	%i7 + 8, %g0 +	 restore %o0, %g0, %o0 +99:	mov	EINVAL, %o0 +98:	call	__errno_location +	 mov	%o0, %i0 +	st	%i0, [%o0] +	jmpl	%i7 + 8, %g0 +	 restore %g0,-1,%o0 +END(__clone) + +	.type __thread_start,@function +__thread_start: +	mov	%g0, %fp	/* terminate backtrace */ +	call	%g2 +	 mov	%g3,%o0 +	call	HIDDEN_JUMPTARGET(_exit),0 +	 nop + +	.size	__thread_start, .-__thread_start + +libc_hidden_def(__clone) +weak_alias(__clone, clone) diff --git a/libc/sysdeps/linux/sparc64/crt1.S b/libc/sysdeps/linux/sparc64/crt1.S new file mode 100644 index 000000000..55e0dea08 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/crt1.S @@ -0,0 +1,116 @@ +/* Startup code for elf64-sparc +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   Contributed by Richard Henderson <richard@gnu.ai.mit.edu>, 1997. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 of the License, or (at your option) any later version. + +   In addition to the permissions in the GNU Lesser General Public +   License, the Free Software Foundation gives you unlimited +   permission to link the compiled version of this file with other +   programs, and to distribute those programs without any restriction +   coming from the use of this file. (The GNU Lesser General Public +   License restrictions do apply in other respects; for example, they +   cover modification of the file, and distribution when not linked +   into another program.) + +   Note that people who make modified versions of this file are not +   obligated to grant this special exception for their modified +   versions; it is their choice whether to do so. The GNU Lesser +   General Public License gives permission to release a modified +   version without this exception; this exception also makes it +   possible to release a modified version which carries forward this +   exception. + +   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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <features.h> +#include <sysdep.h> + +.text +.align 4 +.global _start +.type   _start,%function +#if defined(__UCLIBC_CTOR_DTOR__) +.type   _init,%function +.type   _fini,%function +#else +.weak   _init +.weak   _fini +#endif +.type   __uClibc_main,%function +/* Stick in a dummy reference to main(), so that if an application + * is linking when the main() function is in a static library (.a) + * we can be sure that main() actually gets linked in */ +.global main +.type   main,%function + +#ifdef __PIC__ +.LLGETPC0: +	retl +	add	%o7, %l7, %l7 +#endif + +_start: +#ifdef __PIC__ +	sethi	%hi(_GLOBAL_OFFSET_TABLE_-4), %l7 +	call	.LLGETPC0 +	add	%l7, %lo(_GLOBAL_OFFSET_TABLE_+4), %l7 +#endif + +	/* Terminate the stack frame, and reserve space for functions to +	 * drop their arguments.  */ +	mov	%g0, %fp +	sub	%sp, 6*8, %sp + +  	/* Extract the arguments and environment as encoded on the stack.  The +    	 argument info starts after one register window (16 words) past the SP, +    	 plus the bias we added, plus the magic v9 STACK_BIAS.  */ +	ldx	[%sp+STACK_BIAS+22*8], %o1   /* %o1 = argc */ +	add	%sp, STACK_BIAS+23*8, %o2    /* %o2 = argv */ + +	/* Load the addresses of the user entry points.  */ +	sethi	%hi(main), %o0 +	sethi	%hi(_init), %o3 +	sethi	%hi(_fini), %o4 +	or	%o0, %lo(main), %o0 +	or	%o3, %lo(_init), %o3 +	or	%o4, %lo(_fini), %o4 +#ifdef __PIC__ +	/* Need a little more magic when building PIC to get addr of main */ +	LD	[%l7 + %o0], %o0 +	LD	[%l7 + %o3], %o3 +	LD	[%l7 + %o4], %o4 +#endif + +	/* When starting a binary via the dynamic linker, %g1 contains the +	 * address of the shared library termination function, which will be +	 * registered with atexit().  If we are statically linked, this will +	 * be NULL.  */ +	mov	%g1, %o5 + +	/* Let libc do the rest of the initialization, and call main.  */ +	call __uClibc_main +	nop + +	/* Die very horribly if exit returns.  */ +	unimp + +.size _start,.-_start + +/* Define a symbol for the first piece of initialized data. */ +.data +.global __data_start +__data_start: +.long 0 +.weak data_start +	data_start = __data_start diff --git a/libc/sysdeps/linux/sparc64/crti.S b/libc/sysdeps/linux/sparc64/crti.S new file mode 100644 index 000000000..b7b257dd8 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/crti.S @@ -0,0 +1,23 @@ +	.file	"initfini.c" +	 +	.section .init +	.align 4 +	.global _init +	.type	_init, %function +	.proc	020 +_init: +	!#PROLOGUE# 0 +	save	%sp, -176, %sp +	 +	.align 4 +	 +	 +	.section .fini +	.align 4 +	.global _fini +	.type	_fini, %function +	.proc	020 +_fini: +	!#PROLOGUE# 0 +	save	%sp, -176, %sp +	.align 4 diff --git a/libc/sysdeps/linux/sparc64/crtn.S b/libc/sysdeps/linux/sparc64/crtn.S new file mode 100644 index 000000000..d64ffd020 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/crtn.S @@ -0,0 +1,17 @@ +	.file	"initfini.c" +	 +	.section .init +	.align 4 +	.global _init +	.type	_init, %function +	.proc	020 +	ret +	restore +	 +	.section .fini +	.align 4 +	.global _fini +	.type	_fini, %function +	.proc	020 +	ret +	restore diff --git a/libc/sysdeps/linux/sparc64/fork.S b/libc/sysdeps/linux/sparc64/fork.S new file mode 100644 index 000000000..55c5e8d5c --- /dev/null +++ b/libc/sysdeps/linux/sparc64/fork.S @@ -0,0 +1,32 @@ +/* + * The fork and vfork system calls are special on sparc64: + * they return the "other process" pid in %o0 and the + * "is child" flag in %o1 + */ + +#include <features.h> +#include <sys/syscall.h> + +.text +.global	fork +.type 	fork, @function +.align 4 + +fork: +	mov	__NR_fork, %g1 +	ta	0x6d +	sub	%o1, 1, %o1 +	bcc,pt	%xcc, 1f +	  and	%o0, %o1, %o0 +	sethi	%hi(errno), %g4 +	or	%g4, %lo(errno), %g4 +	st	%o0, [%g4] +	retl +	  mov	-1, %o0 +1: +       	retl +	  nop + +.size fork, . - fork + +libc_hidden_def(fork) diff --git a/libc/sysdeps/linux/sparc64/fpu_control.h b/libc/sysdeps/linux/sparc64/fpu_control.h new file mode 100644 index 000000000..77acde925 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/fpu_control.h @@ -0,0 +1,72 @@ +/* FPU control word bits.  SPARC version. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Miguel de Icaza + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _FPU_CONTROL_H +#define _FPU_CONTROL_H	1 + + +#include <features.h> +#include <bits/wordsize.h> + +/* masking of interrupts */ +#define _FPU_MASK_IM  0x08000000 +#define _FPU_MASK_OM  0x04000000 +#define _FPU_MASK_UM  0x02000000 +#define _FPU_MASK_ZM  0x01000000 +#define _FPU_MASK_PM  0x00800000 + +/* precision control */ +#define _FPU_EXTENDED 0x00000000     /* RECOMMENDED */ +#define _FPU_DOUBLE   0x20000000 +#define _FPU_80BIT    0x30000000 +#define _FPU_SINGLE   0x10000000     /* DO NOT USE */ + +/* rounding control / Sparc */ +#define _FPU_RC_DOWN    0xc0000000 +#define _FPU_RC_UP      0x80000000 +#define _FPU_RC_ZERO    0x40000000 +#define _FPU_RC_NEAREST 0x0        /* RECOMMENDED */ + +#define _FPU_RESERVED   0x30300000  /* Reserved bits in cw */ + + +/* Now two recommended cw */ + +/* Linux and IEEE default: +     - extended precision +     - rounding to nearest +     - no exceptions  */ +#define _FPU_DEFAULT  0x0 +#define _FPU_IEEE     0x0 + +/* Type of the control word.  */ +typedef unsigned long int fpu_control_t; + +#if __WORDSIZE == 64 +# define _FPU_GETCW(cw) __asm__ __volatile__ ("stx %%fsr,%0" : "=m" (*&cw)) +# define _FPU_SETCW(cw) __asm__ __volatile__ ("ldx %0,%%fsr" : : "m" (*&cw)) +#else +# define _FPU_GETCW(cw) __asm__ __volatile__ ("st %%fsr,%0" : "=m" (*&cw)) +# define _FPU_SETCW(cw) __asm__ __volatile__ ("ld %0,%%fsr" : : "m" (*&cw)) +#endif + +/* Default control word set at startup.  */ +extern fpu_control_t __fpu_control; + +#endif	/* fpu_control.h */ diff --git a/libc/sysdeps/linux/sparc64/getcontext.S b/libc/sysdeps/linux/sparc64/getcontext.S new file mode 100644 index 000000000..44b716d6a --- /dev/null +++ b/libc/sysdeps/linux/sparc64/getcontext.S @@ -0,0 +1,63 @@ +/* Copyright (C) 1997-2017 Free Software Foundation, Inc. +   Contributed by Richard Henderson (rth@tamu.edu). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <sysdep.h> +#include "ucontext_i.h" + +/* int getcontext(ucontext_t *); */ + +ENTRY(__getcontext) + +	ldx	[%o0 + UC_LINK], %o1	/* Preserve uc_link field, the +					   trap clears it.  */ +	ta	0x6e +1: +	ldx	[%o0 + UC_M_PC], %o2 +	ldx	[%o0 + UC_M_NPC], %o3 +	ldx	[%o0 + __UC_SIGMASK], %o4 +	stx	%o1, [%o0 + UC_LINK] +	add	%o2, 2f - 1b, %o2 +	stx	%o2, [%o0 + UC_M_PC] +	add	%o3, 2f - 1b, %o3 +	stx	%o3, [%o0 + UC_M_NPC] +#if SIGMASK_WORDS == 16 +	stx	%o4, [%o0 + UC_SIGMASK] +	stx	%g0, [%o0 + UC_SIGMASK + 8] +	stx	%g0, [%o0 + UC_SIGMASK + 16] +	stx	%g0, [%o0 + UC_SIGMASK + 24] +	stx	%g0, [%o0 + UC_SIGMASK + 32] +	stx	%g0, [%o0 + UC_SIGMASK + 40] +	stx	%g0, [%o0 + UC_SIGMASK + 48] +	stx	%g0, [%o0 + UC_SIGMASK + 56] +	stx	%g0, [%o0 + UC_SIGMASK + 64] +	stx	%g0, [%o0 + UC_SIGMASK + 72] +	stx	%g0, [%o0 + UC_SIGMASK + 80] +	stx	%g0, [%o0 + UC_SIGMASK + 88] +	stx	%g0, [%o0 + UC_SIGMASK + 96] +	stx	%g0, [%o0 + UC_SIGMASK + 104] +	stx	%g0, [%o0 + UC_SIGMASK + 112] +	stx	%g0, [%o0 + UC_SIGMASK + 120] +#else +# error Adjust __getcontext +#endif +2: +	retl +	 clr	%o0 + +END(__getcontext) + +weak_alias (__getcontext, getcontext) diff --git a/libc/sysdeps/linux/sparc64/jmpbuf-offsets.h b/libc/sysdeps/linux/sparc64/jmpbuf-offsets.h new file mode 100644 index 000000000..e273f5561 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/jmpbuf-offsets.h @@ -0,0 +1,20 @@ +/* Private macros for accessing __jmp_buf contents.  SPARC version. +   Copyright (C) 2006 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#define O_mask_was_saved	512 +#define O_gregs			32 +#define O_g1			(O_gregs + 4*8) diff --git a/libc/sysdeps/linux/sparc64/jmpbuf-unwind.h b/libc/sysdeps/linux/sparc64/jmpbuf-unwind.h new file mode 100644 index 000000000..31841e827 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/jmpbuf-unwind.h @@ -0,0 +1,28 @@ +/* + * Copyright (C) 2000-2006 Erik Andersen <andersen@uclibc.org> + * + * Licensed under the LGPL v2.1, see the file COPYING.LIB in this tarball. + */ + +#include <setjmp.h> +#include <jmpbuf-offsets.h> + +/* Test if longjmp to JMPBUF would unwind the frame +   containing a local variable at ADDRESS.  */ +#define _JMPBUF_UNWINDS(jmpbuf, address)		\ +  ((unsigned long int) (address)			\ +   < (jmpbuf)->__uc_mcontext.__mc_gregs[MC_O6] + 2047) + +#ifdef __UCLIBC_HAS_THREADS_NATIVE__ +#include <setjmp.h> +#include <stdint.h> +#include <unwind.h> + +#define _JMPBUF_CFA_UNWINDS_ADJ(_jmpbuf, _context, _adj) \ +  _JMPBUF_UNWINDS_ADJ (_jmpbuf, (void *) _Unwind_GetCFA (_context), _adj) + +#define _JMPBUF_UNWINDS_ADJ(_jmpbuf, _address, _adj) \ +  ((uintptr_t) (_address) - (_adj) \ +   < (uintptr_t) (_jmpbuf)[0].__uc_mcontext.__mc_gregs[MC_O6] + 2047 - (_adj)) + +#endif diff --git a/libc/sysdeps/linux/sparc64/makecontext.c b/libc/sysdeps/linux/sparc64/makecontext.c new file mode 100644 index 000000000..3750f012c --- /dev/null +++ b/libc/sysdeps/linux/sparc64/makecontext.c @@ -0,0 +1,56 @@ +/* Copyright (C) 2001-2017 Free Software Foundation, Inc. +   Contributed by Jakub Jelinek <jakub@redhat.com>. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <stdarg.h> +#include <stdio.h> +#include <stdlib.h> +#include <ucontext.h> + +extern void __start_context (struct ucontext *ucp); + +void +__makecontext (ucontext_t *ucp, void (*func) (void), int argc, ...) +{ +  extern void __makecontext_ret (void); +  unsigned long *sp, *topsp; +  va_list ap; +  int i; + +  sp = (unsigned long *) ((long) ucp->uc_stack.ss_sp + ucp->uc_stack.ss_size); +  sp -= (argc > 6 ? argc : 6) + 32; +  sp = (unsigned long *) (((long) sp) & -16L); +  topsp = sp + (argc > 6 ? argc : 6) + 16; + +  ucp->uc_mcontext.mc_gregs[MC_PC] = (long) func; +  ucp->uc_mcontext.mc_gregs[MC_NPC] = ((long) func) + 4; +  ucp->uc_mcontext.mc_gregs[MC_O6] = ((long) sp) - 0x7ff; +  ucp->uc_mcontext.mc_gregs[MC_O7] = ((long) __start_context) - 8; +  ucp->uc_mcontext.mc_fp = ((long) topsp) - 0x7ff; +  ucp->uc_mcontext.mc_i7 = 0; +  topsp[14] = 0; +  topsp[15] = 0; +  sp[8] = (long) ucp->uc_link; +  va_start (ap, argc); +  for (i = 0; i < argc; ++i) +    if (i < 6) +      ucp->uc_mcontext.mc_gregs[MC_O0 + i] = va_arg (ap, long); +    else +      sp[16 + i] = va_arg (ap, long); +  va_end (ap); +} + +weak_alias (__makecontext, makecontext) diff --git a/libc/sysdeps/linux/sparc64/pipe.S b/libc/sysdeps/linux/sparc64/pipe.S new file mode 100644 index 000000000..b04c7b104 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/pipe.S @@ -0,0 +1,38 @@ +/* Copyright (C) 1997-2017 Free Software Foundation, Inc. +   Contributed by Miguel de Icaza <miguel@gnu.ai.mit.edu>, 1997. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <sysdep.h> + +	.text + +	.globl	__syscall_error +ENTRY(__pipe) +	mov	%o0, %o2		/* Save PIPEDES. */ +	LOADSYSCALL(pipe) +	ta	0x6d +	bcc,pt	%xcc, 1f +	 mov	%o7, %g1 +	call	__syscall_error +	 mov	%g1, %o7 +1:	st	%o0, [%o2]		/* PIPEDES[0] = %o0; */ +	st	%o1, [%o2 + 4]		/* PIPEDES[1] = %o1; */ +	retl +	 clr	%o0 +END(__pipe) + +weak_alias(__pipe, pipe) +libc_hidden_def(pipe) diff --git a/libc/sysdeps/linux/sparc64/setcontext.S b/libc/sysdeps/linux/sparc64/setcontext.S new file mode 100644 index 000000000..a899549a8 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/setcontext.S @@ -0,0 +1,37 @@ +/* Copyright (C) 1997-2017 Free Software Foundation, Inc. +   Contributed by Richard Henderson (rth@tamu.edu). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <sysdep.h> +#include "ucontext_i.h" + +/* int setcontext(ucontext_t *ctx); */ +.weak setcontext +ENTRY(setcontext) + +	ba,pt	%xcc, 1f +	 mov	1, %o1 + +END(setcontext) + +/* int __setcontext(ucontext_t *ctx, int restoremask); */ +ENTRY(__setcontext) + +1:	ldx	[%o0 + UC_SIGMASK], %o2 +	stx	%o2, [%o0 + __UC_SIGMASK] +	ta	0x6f + +END(__setcontext) diff --git a/libc/sysdeps/linux/sparc64/setjmp.S b/libc/sysdeps/linux/sparc64/setjmp.S new file mode 100644 index 000000000..b30c7d744 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/setjmp.S @@ -0,0 +1,63 @@ +/* Copyright (C) 1997-2017 Free Software Foundation, Inc. +   Contributed by Richard Henderson (rth@tamu.edu). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +/* __sigsetjmp is implemented in terms of the getcontext trap on +   Linux/Sparc64.  */ + +#include <sysdep.h> + +/* Offsets into the jmp_buf structure.  */ + +#define O_mask_was_saved	512 +#define O_gregs			32 +#define O_g1			(O_gregs + 4*8) + +/* int _setjmp(jmp_buf) */ + +ENTRY(_setjmp) +	ba	__sigsetjmp_local +	 set	0, %o1 +END(_setjmp) +libc_hidden_def(_setjmp) + +/* int setjmp(jmp_buf) */ + +ENTRY(setjmp) +	ba,pt	%xcc, __sigsetjmp_local +	 set	1, %o1 +END(setjmp) + +/* int __sigsetjmp(jmp_buf, savemask)  */ + +ENTRY(__sigsetjmp) +__sigsetjmp_local: + +	/* Record whether the user is intending to save the sigmask.  */ +	st	%o1, [%o0 + O_mask_was_saved] + +	/* Load up our return value, as longjmp is going to override +	   the jmp_buf on its way back.  */ +	mov	%g0, %g1 + +	/* And call getcontext!  */ +	ta	0x6e + +	retl +	 mov	%g1, %o0 + +END(__sigsetjmp) +libc_hidden_def(__sigsetjmp) diff --git a/libc/sysdeps/linux/sparc64/sigaction.c b/libc/sysdeps/linux/sparc64/sigaction.c new file mode 100644 index 000000000..d8aaad0fb --- /dev/null +++ b/libc/sysdeps/linux/sparc64/sigaction.c @@ -0,0 +1,77 @@ +/* POSIX.1 sigaction call for Linux/SPARC64. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   Contributed by Miguel de Icaza <miguel@nuclecu.unam.mx> and +		  Jakub Jelinek <jj@ultra.linux.cz>. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <string.h> +#include <syscall.h> +#include <sysdep.h> +#include <sys/signal.h> +#include <errno.h> +#include <bits/kernel_sigaction.h> + +/* SPARC 64bit userland requires a kernel that has rt signals anyway. */ + +static void __rt_sigreturn_stub (void); + +int __libc_sigaction (int sig, const struct sigaction *act, struct sigaction *oact) +{ +  int ret; +  struct kernel_sigaction kact, koact; +  unsigned long stub = ((unsigned long) &__rt_sigreturn_stub) - 8; + +  if (act) { +      kact.k_sa_handler = act->sa_handler; +      memcpy (&kact.sa_mask, &act->sa_mask, sizeof (sigset_t)); +      kact.sa_flags = act->sa_flags; +      kact.sa_restorer = NULL; +  } + +  /* XXX The size argument hopefully will have to be changed to the +     real size of the user-level sigset_t.  */ +  ret = INLINE_SYSCALL (rt_sigaction, 5, sig, +			act ? &kact : 0, +			oact ? &koact : 0, stub, _NSIG / 8); + +  if (oact && ret >= 0) { +      oact->sa_handler = koact.k_sa_handler; +      memcpy (&oact->sa_mask, &koact.sa_mask, sizeof (sigset_t)); +      oact->sa_flags = koact.sa_flags; +      oact->sa_restorer = koact.sa_restorer; +  } + +  return ret; +} + +#ifndef LIBC_SIGACTION +# ifndef __UCLIBC_HAS_THREADS__ +strong_alias(__libc_sigaction,sigaction) +libc_hidden_def(sigaction) +# else +weak_alias(__libc_sigaction,sigaction) +libc_hidden_weak(sigaction) +# endif +#endif + +static void +__rt_sigreturn_stub (void) +{ +  __asm__ ("mov %0, %%g1\n\t" +	   "ta	0x6d\n\t" +	   : /* no outputs */ +	   : "i" (__NR_rt_sigreturn)); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/Makefile b/libc/sysdeps/linux/sparc64/soft-fp/Makefile new file mode 100644 index 000000000..b145df283 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/Makefile @@ -0,0 +1,33 @@ +#  Software floating-point emulation. +#  Makefile for SPARC v9 ABI mandated long double utility +#  functions (_Qp_*). +#  Copyright (C) 1999-2017 Free Software Foundation, Inc. +#  This file is part of the GNU C Library. +#  Contributed by Jakub Jelinek (jj@ultra.linux.cz). +# + +# The GNU C Library is free software; you can redistribute it and/or +# modify it under the terms of the GNU Lesser General Public +# License as published by the Free Software Foundation; either +# version 2.1 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 +# Lesser General Public License for more details. + +# You should have received a copy of the GNU Lesser General Public +# License along with the GNU C Library; if not, see +# <http://www.gnu.org/licenses/>. + +ifeq ($(subdir),soft-fp) +sparc64-quad-routines := qp_add qp_cmp qp_cmpe qp_div qp_dtoq qp_feq qp_fge \ +	qp_fgt qp_fle qp_flt qp_fne qp_itoq qp_mul qp_neg qp_qtod qp_qtoi   \ +	qp_qtos qp_qtoui qp_qtoux qp_qtox qp_sqrt qp_stoq qp_sub qp_uitoq   \ +	qp_uxtoq qp_xtoq qp_util +sysdep_routines += $(sparc64-quad-routines) +endif + +ifeq ($(subdir),math) +CPPFLAGS += -I../soft-fp/ +endif diff --git a/libc/sysdeps/linux/sparc64/soft-fp/Versions b/libc/sysdeps/linux/sparc64/soft-fp/Versions new file mode 100644 index 000000000..9e89c3c3e --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/Versions @@ -0,0 +1,8 @@ +libc { +  GLIBC_2.2 { +    _Qp_add; _Qp_cmp; _Qp_cmpe; _Qp_div; _Qp_dtoq; _Qp_feq; _Qp_fge; _Qp_fgt; +    _Qp_fle; _Qp_flt; _Qp_fne; _Qp_itoq; _Qp_mul; _Qp_neg; _Qp_qtod; _Qp_qtoi; +    _Qp_qtos; _Qp_qtoui; _Qp_qtoux; _Qp_qtox; _Qp_sqrt; _Qp_stoq; _Qp_sub; +    _Qp_uitoq; _Qp_uxtoq; _Qp_xtoq; +  } +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/double.h b/libc/sysdeps/linux/sparc64/soft-fp/double.h new file mode 100644 index 000000000..055d9da6b --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/double.h @@ -0,0 +1,323 @@ +/* Software floating-point emulation. +   Definitions for IEEE Double Precision +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com), +		  Jakub Jelinek (jj@ultra.linux.cz), +		  David S. Miller (davem@redhat.com) and +		  Peter Maydell (pmaydell@chiark.greenend.org.uk). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 of the License, or (at your option) any later version. + +   In addition to the permissions in the GNU Lesser General Public +   License, the Free Software Foundation gives you unlimited +   permission to link the compiled version of this file into +   combinations with other programs, and to distribute those +   combinations without any restriction coming from the use of this +   file.  (The Lesser General Public License restrictions do apply in +   other respects; for example, they cover modification of the file, +   and distribution when not linked into a combine executable.) + +   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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef SOFT_FP_DOUBLE_H +#define SOFT_FP_DOUBLE_H	1 + +#if _FP_W_TYPE_SIZE < 32 +# error "Here's a nickel kid.  Go buy yourself a real computer." +#endif + +#if _FP_W_TYPE_SIZE < 64 +# define _FP_FRACTBITS_D	(2 * _FP_W_TYPE_SIZE) +# define _FP_FRACTBITS_DW_D	(4 * _FP_W_TYPE_SIZE) +#else +# define _FP_FRACTBITS_D	_FP_W_TYPE_SIZE +# define _FP_FRACTBITS_DW_D	(2 * _FP_W_TYPE_SIZE) +#endif + +#define _FP_FRACBITS_D		53 +#define _FP_FRACXBITS_D		(_FP_FRACTBITS_D - _FP_FRACBITS_D) +#define _FP_WFRACBITS_D		(_FP_WORKBITS + _FP_FRACBITS_D) +#define _FP_WFRACXBITS_D	(_FP_FRACTBITS_D - _FP_WFRACBITS_D) +#define _FP_EXPBITS_D		11 +#define _FP_EXPBIAS_D		1023 +#define _FP_EXPMAX_D		2047 + +#define _FP_QNANBIT_D		\ +	((_FP_W_TYPE) 1 << (_FP_FRACBITS_D-2) % _FP_W_TYPE_SIZE) +#define _FP_QNANBIT_SH_D		\ +	((_FP_W_TYPE) 1 << (_FP_FRACBITS_D-2+_FP_WORKBITS) % _FP_W_TYPE_SIZE) +#define _FP_IMPLBIT_D		\ +	((_FP_W_TYPE) 1 << (_FP_FRACBITS_D-1) % _FP_W_TYPE_SIZE) +#define _FP_IMPLBIT_SH_D		\ +	((_FP_W_TYPE) 1 << (_FP_FRACBITS_D-1+_FP_WORKBITS) % _FP_W_TYPE_SIZE) +#define _FP_OVERFLOW_D		\ +	((_FP_W_TYPE) 1 << _FP_WFRACBITS_D % _FP_W_TYPE_SIZE) + +#define _FP_WFRACBITS_DW_D	(2 * _FP_WFRACBITS_D) +#define _FP_WFRACXBITS_DW_D	(_FP_FRACTBITS_DW_D - _FP_WFRACBITS_DW_D) +#define _FP_HIGHBIT_DW_D	\ +  ((_FP_W_TYPE) 1 << (_FP_WFRACBITS_DW_D - 1) % _FP_W_TYPE_SIZE) + +typedef float DFtype __attribute__ ((mode (DF))); + +#if _FP_W_TYPE_SIZE < 64 + +union _FP_UNION_D +{ +  DFtype flt; +  struct _FP_STRUCT_LAYOUT +  { +# if __BYTE_ORDER == __BIG_ENDIAN +    unsigned sign  : 1; +    unsigned exp   : _FP_EXPBITS_D; +    unsigned frac1 : _FP_FRACBITS_D - (_FP_IMPLBIT_D != 0) - _FP_W_TYPE_SIZE; +    unsigned frac0 : _FP_W_TYPE_SIZE; +# else +    unsigned frac0 : _FP_W_TYPE_SIZE; +    unsigned frac1 : _FP_FRACBITS_D - (_FP_IMPLBIT_D != 0) - _FP_W_TYPE_SIZE; +    unsigned exp   : _FP_EXPBITS_D; +    unsigned sign  : 1; +# endif +  } bits __attribute__ ((packed)); +}; + +# define FP_DECL_D(X)		_FP_DECL (2, X) +# define FP_UNPACK_RAW_D(X, val)	_FP_UNPACK_RAW_2 (D, X, (val)) +# define FP_UNPACK_RAW_DP(X, val)	_FP_UNPACK_RAW_2_P (D, X, (val)) +# define FP_PACK_RAW_D(val, X)	_FP_PACK_RAW_2 (D, (val), X) +# define FP_PACK_RAW_DP(val, X)			\ +  do						\ +    {						\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_2_P (D, (val), X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_D(X, val)			\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_2 (D, X, (val));		\ +      _FP_UNPACK_CANONICAL (D, 2, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_DP(X, val)			\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_2_P (D, X, (val));		\ +      _FP_UNPACK_CANONICAL (D, 2, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_SEMIRAW_D(X, val)		\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_2 (D, X, (val));		\ +      _FP_UNPACK_SEMIRAW (D, 2, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_SEMIRAW_DP(X, val)		\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_2_P (D, X, (val));		\ +      _FP_UNPACK_SEMIRAW (D, 2, X);		\ +    }						\ +  while (0) + +# define FP_PACK_D(val, X)			\ +  do						\ +    {						\ +      _FP_PACK_CANONICAL (D, 2, X);		\ +      _FP_PACK_RAW_2 (D, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_DP(val, X)			\ +  do						\ +    {						\ +      _FP_PACK_CANONICAL (D, 2, X);		\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_2_P (D, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_SEMIRAW_D(val, X)		\ +  do						\ +    {						\ +      _FP_PACK_SEMIRAW (D, 2, X);		\ +      _FP_PACK_RAW_2 (D, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_SEMIRAW_DP(val, X)		\ +  do						\ +    {						\ +      _FP_PACK_SEMIRAW (D, 2, X);		\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_2_P (D, (val), X);		\ +    }						\ +  while (0) + +# define FP_ISSIGNAN_D(X)		_FP_ISSIGNAN (D, 2, X) +# define FP_NEG_D(R, X)			_FP_NEG (D, 2, R, X) +# define FP_ADD_D(R, X, Y)		_FP_ADD (D, 2, R, X, Y) +# define FP_SUB_D(R, X, Y)		_FP_SUB (D, 2, R, X, Y) +# define FP_MUL_D(R, X, Y)		_FP_MUL (D, 2, R, X, Y) +# define FP_DIV_D(R, X, Y)		_FP_DIV (D, 2, R, X, Y) +# define FP_SQRT_D(R, X)		_FP_SQRT (D, 2, R, X) +# define _FP_SQRT_MEAT_D(R, S, T, X, Q)	_FP_SQRT_MEAT_2 (R, S, T, X, (Q)) +# define FP_FMA_D(R, X, Y, Z)		_FP_FMA (D, 2, 4, R, X, Y, Z) + +# define FP_CMP_D(r, X, Y, un, ex)	_FP_CMP (D, 2, (r), X, Y, (un), (ex)) +# define FP_CMP_EQ_D(r, X, Y, ex)	_FP_CMP_EQ (D, 2, (r), X, Y, (ex)) +# define FP_CMP_UNORD_D(r, X, Y, ex)	_FP_CMP_UNORD (D, 2, (r), X, Y, (ex)) + +# define FP_TO_INT_D(r, X, rsz, rsg)	_FP_TO_INT (D, 2, (r), X, (rsz), (rsg)) +# define FP_TO_INT_ROUND_D(r, X, rsz, rsg)	\ +  _FP_TO_INT_ROUND (D, 2, (r), X, (rsz), (rsg)) +# define FP_FROM_INT_D(X, r, rs, rt)	_FP_FROM_INT (D, 2, X, (r), (rs), rt) + +# define _FP_FRAC_HIGH_D(X)	_FP_FRAC_HIGH_2 (X) +# define _FP_FRAC_HIGH_RAW_D(X)	_FP_FRAC_HIGH_2 (X) + +# define _FP_FRAC_HIGH_DW_D(X)	_FP_FRAC_HIGH_4 (X) + +#else + +union _FP_UNION_D +{ +  DFtype flt; +  struct _FP_STRUCT_LAYOUT +  { +# if __BYTE_ORDER == __BIG_ENDIAN +    unsigned sign   : 1; +    unsigned exp    : _FP_EXPBITS_D; +    _FP_W_TYPE frac : _FP_FRACBITS_D - (_FP_IMPLBIT_D != 0); +# else +    _FP_W_TYPE frac : _FP_FRACBITS_D - (_FP_IMPLBIT_D != 0); +    unsigned exp    : _FP_EXPBITS_D; +    unsigned sign   : 1; +# endif +  } bits __attribute__ ((packed)); +}; + +# define FP_DECL_D(X)		_FP_DECL (1, X) +# define FP_UNPACK_RAW_D(X, val)	_FP_UNPACK_RAW_1 (D, X, (val)) +# define FP_UNPACK_RAW_DP(X, val)	_FP_UNPACK_RAW_1_P (D, X, (val)) +# define FP_PACK_RAW_D(val, X)	_FP_PACK_RAW_1 (D, (val), X) +# define FP_PACK_RAW_DP(val, X)			\ +  do						\ +    {						\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_1_P (D, (val), X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_D(X, val)			\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_1 (D, X, (val));		\ +      _FP_UNPACK_CANONICAL (D, 1, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_DP(X, val)			\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_1_P (D, X, (val));		\ +      _FP_UNPACK_CANONICAL (D, 1, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_SEMIRAW_D(X, val)		\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_1 (D, X, (val));		\ +      _FP_UNPACK_SEMIRAW (D, 1, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_SEMIRAW_DP(X, val)		\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_1_P (D, X, (val));		\ +      _FP_UNPACK_SEMIRAW (D, 1, X);		\ +    }						\ +  while (0) + +# define FP_PACK_D(val, X)			\ +  do						\ +    {						\ +      _FP_PACK_CANONICAL (D, 1, X);		\ +      _FP_PACK_RAW_1 (D, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_DP(val, X)			\ +  do						\ +    {						\ +      _FP_PACK_CANONICAL (D, 1, X);		\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_1_P (D, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_SEMIRAW_D(val, X)		\ +  do						\ +    {						\ +      _FP_PACK_SEMIRAW (D, 1, X);		\ +      _FP_PACK_RAW_1 (D, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_SEMIRAW_DP(val, X)		\ +  do						\ +    {						\ +      _FP_PACK_SEMIRAW (D, 1, X);		\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_1_P (D, (val), X);		\ +    }						\ +  while (0) + +# define FP_ISSIGNAN_D(X)		_FP_ISSIGNAN (D, 1, X) +# define FP_NEG_D(R, X)			_FP_NEG (D, 1, R, X) +# define FP_ADD_D(R, X, Y)		_FP_ADD (D, 1, R, X, Y) +# define FP_SUB_D(R, X, Y)		_FP_SUB (D, 1, R, X, Y) +# define FP_MUL_D(R, X, Y)		_FP_MUL (D, 1, R, X, Y) +# define FP_DIV_D(R, X, Y)		_FP_DIV (D, 1, R, X, Y) +# define FP_SQRT_D(R, X)		_FP_SQRT (D, 1, R, X) +# define _FP_SQRT_MEAT_D(R, S, T, X, Q)	_FP_SQRT_MEAT_1 (R, S, T, X, (Q)) +# define FP_FMA_D(R, X, Y, Z)		_FP_FMA (D, 1, 2, R, X, Y, Z) + +/* The implementation of _FP_MUL_D and _FP_DIV_D should be chosen by +   the target machine.  */ + +# define FP_CMP_D(r, X, Y, un, ex)	_FP_CMP (D, 1, (r), X, Y, (un), (ex)) +# define FP_CMP_EQ_D(r, X, Y, ex)	_FP_CMP_EQ (D, 1, (r), X, Y, (ex)) +# define FP_CMP_UNORD_D(r, X, Y, ex)	_FP_CMP_UNORD (D, 1, (r), X, Y, (ex)) + +# define FP_TO_INT_D(r, X, rsz, rsg)	_FP_TO_INT (D, 1, (r), X, (rsz), (rsg)) +# define FP_TO_INT_ROUND_D(r, X, rsz, rsg)	\ +  _FP_TO_INT_ROUND (D, 1, (r), X, (rsz), (rsg)) +# define FP_FROM_INT_D(X, r, rs, rt)	_FP_FROM_INT (D, 1, X, (r), (rs), rt) + +# define _FP_FRAC_HIGH_D(X)	_FP_FRAC_HIGH_1 (X) +# define _FP_FRAC_HIGH_RAW_D(X)	_FP_FRAC_HIGH_1 (X) + +# define _FP_FRAC_HIGH_DW_D(X)	_FP_FRAC_HIGH_2 (X) + +#endif /* W_TYPE_SIZE < 64 */ + +#endif /* !SOFT_FP_DOUBLE_H */ diff --git a/libc/sysdeps/linux/sparc64/soft-fp/e_ilogbl.c b/libc/sysdeps/linux/sparc64/soft-fp/e_ilogbl.c new file mode 100644 index 000000000..5b19d12a4 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/e_ilogbl.c @@ -0,0 +1,79 @@ +/* Software floating-point emulation. +   ilogbl(x, exp) +   Copyright (C) 1999-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +/* ilogbl(long double x) + * return the binary exponent of non-zero x + * ilogbl(0) = 0x80000001 + * ilogbl(inf/NaN) = 0x7fffffff (no signal is raised) + */ + +#include "soft-fp.h" +#include "quad.h" +#include <math.h> + +int __ieee754_ilogbl (long double x) +{ +  FP_DECL_EX; +  FP_DECL_Q(X); + +/* +  FP_UNPACK_Q(X, x); +  switch (X_c) +    { +    case FP_CLS_ZERO: +      return FP_ILOGB0; +    case FP_CLS_NAN: +    case FP_CLS_INF: +      return FP_ILOGBNAN; +    default: +      return X_e; +    } + */ +  FP_UNPACK_RAW_Q(X, x); +  switch (X_e) +    { +    default: +      return X_e - _FP_EXPBIAS_Q; +    case 0: +#if (2 * _FP_W_TYPE_SIZE) < _FP_FRACBITS_Q +      if (_FP_FRAC_ZEROP_4(X)) +	return FP_ILOGB0; +      else +	{ +	  _FP_I_TYPE shift; +	  _FP_FRAC_CLZ_4(shift, X); +	  shift -= _FP_FRACXBITS_Q; +	  return X_e - _FP_EXPBIAS_Q - 1 + shift; +	} +#else +      if (_FP_FRAC_ZEROP_2(X)) +	return FP_ILOGB0; +      else +	{ +	  _FP_I_TYPE shift; +	  _FP_FRAC_CLZ_2(shift, X); +	  shift -= _FP_FRACXBITS_Q; +	  return X_e - _FP_EXPBIAS_Q - 1 + shift; +	} +#endif +    case _FP_EXPBIAS_Q: +      return FP_ILOGBNAN; +    } +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/longlong.h b/libc/sysdeps/linux/sparc64/soft-fp/longlong.h new file mode 100644 index 000000000..0ec11c505 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/longlong.h @@ -0,0 +1,1773 @@ +/* longlong.h -- definitions for mixed size 32/64 bit arithmetic. +   Copyright (C) 1991-2017 Free Software Foundation, Inc. + +   This file is part of the GNU C Library. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 of the License, or (at your option) any later version. + +   In addition to the permissions in the GNU Lesser General Public +   License, the Free Software Foundation gives you unlimited +   permission to link the compiled version of this file into +   combinations with other programs, and to distribute those +   combinations without any restriction coming from the use of this +   file.  (The Lesser General Public License restrictions do apply in +   other respects; for example, they cover modification of the file, +   and distribution when not linked into a combine executable.) + +   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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +/* You have to define the following before including this file: + +   UWtype -- An unsigned type, default type for operations (typically a "word") +   UHWtype -- An unsigned type, at least half the size of UWtype. +   UDWtype -- An unsigned type, at least twice as large a UWtype +   W_TYPE_SIZE -- size in bits of UWtype + +   UQItype -- Unsigned 8 bit type. +   SItype, USItype -- Signed and unsigned 32 bit types. +   DItype, UDItype -- Signed and unsigned 64 bit types. + +   On a 32 bit machine UWtype should typically be USItype; +   on a 64 bit machine, UWtype should typically be UDItype.  */ + +#define __BITS4 (W_TYPE_SIZE / 4) +#define __ll_B ((UWtype) 1 << (W_TYPE_SIZE / 2)) +#define __ll_lowpart(t) ((UWtype) (t) & (__ll_B - 1)) +#define __ll_highpart(t) ((UWtype) (t) >> (W_TYPE_SIZE / 2)) + +#ifndef W_TYPE_SIZE +#define W_TYPE_SIZE	32 +#define UWtype		USItype +#define UHWtype		USItype +#define UDWtype		UDItype +#endif + +/* Used in glibc only.  */ +#ifndef attribute_hidden +#define attribute_hidden +#endif + +extern const UQItype __clz_tab[256] attribute_hidden; + +/* Define auxiliary asm macros. + +   1) umul_ppmm(high_prod, low_prod, multiplier, multiplicand) multiplies two +   UWtype integers MULTIPLIER and MULTIPLICAND, and generates a two UWtype +   word product in HIGH_PROD and LOW_PROD. + +   2) __umulsidi3(a,b) multiplies two UWtype integers A and B, and returns a +   UDWtype product.  This is just a variant of umul_ppmm. + +   3) udiv_qrnnd(quotient, remainder, high_numerator, low_numerator, +   denominator) divides a UDWtype, composed by the UWtype integers +   HIGH_NUMERATOR and LOW_NUMERATOR, by DENOMINATOR and places the quotient +   in QUOTIENT and the remainder in REMAINDER.  HIGH_NUMERATOR must be less +   than DENOMINATOR for correct operation.  If, in addition, the most +   significant bit of DENOMINATOR must be 1, then the pre-processor symbol +   UDIV_NEEDS_NORMALIZATION is defined to 1. + +   4) sdiv_qrnnd(quotient, remainder, high_numerator, low_numerator, +   denominator).  Like udiv_qrnnd but the numbers are signed.  The quotient +   is rounded towards 0. + +   5) count_leading_zeros(count, x) counts the number of zero-bits from the +   msb to the first nonzero bit in the UWtype X.  This is the number of +   steps X needs to be shifted left to set the msb.  Undefined for X == 0, +   unless the symbol COUNT_LEADING_ZEROS_0 is defined to some value. + +   6) count_trailing_zeros(count, x) like count_leading_zeros, but counts +   from the least significant end. + +   7) add_ssaaaa(high_sum, low_sum, high_addend_1, low_addend_1, +   high_addend_2, low_addend_2) adds two UWtype integers, composed by +   HIGH_ADDEND_1 and LOW_ADDEND_1, and HIGH_ADDEND_2 and LOW_ADDEND_2 +   respectively.  The result is placed in HIGH_SUM and LOW_SUM.  Overflow +   (i.e. carry out) is not stored anywhere, and is lost. + +   8) sub_ddmmss(high_difference, low_difference, high_minuend, low_minuend, +   high_subtrahend, low_subtrahend) subtracts two two-word UWtype integers, +   composed by HIGH_MINUEND_1 and LOW_MINUEND_1, and HIGH_SUBTRAHEND_2 and +   LOW_SUBTRAHEND_2 respectively.  The result is placed in HIGH_DIFFERENCE +   and LOW_DIFFERENCE.  Overflow (i.e. carry out) is not stored anywhere, +   and is lost. + +   If any of these macros are left undefined for a particular CPU, +   C macros are used.  */ + +/* The CPUs come in alphabetical order below. + +   Please add support for more CPUs here, or improve the current support +   for the CPUs below! +   (E.g. WE32100, IBM360.)  */ + +#if defined (__GNUC__) && !defined (NO_ASM) + +/* We sometimes need to clobber "cc" with gcc2, but that would not be +   understood by gcc1.  Use cpp to avoid major code duplication.  */ +#if __GNUC__ < 2 +#define __CLOBBER_CC +#define __AND_CLOBBER_CC +#else /* __GNUC__ >= 2 */ +#define __CLOBBER_CC : "cc" +#define __AND_CLOBBER_CC , "cc" +#endif /* __GNUC__ < 2 */ + +#if defined (__aarch64__) + +#if W_TYPE_SIZE == 32 +#define count_leading_zeros(COUNT, X)	((COUNT) = __builtin_clz (X)) +#define count_trailing_zeros(COUNT, X)   ((COUNT) = __builtin_ctz (X)) +#define COUNT_LEADING_ZEROS_0 32 +#endif /* W_TYPE_SIZE == 32 */ + +#if W_TYPE_SIZE == 64 +#define count_leading_zeros(COUNT, X)	((COUNT) = __builtin_clzll (X)) +#define count_trailing_zeros(COUNT, X)   ((COUNT) = __builtin_ctzll (X)) +#define COUNT_LEADING_ZEROS_0 64 +#endif /* W_TYPE_SIZE == 64 */ + +#endif /* __aarch64__ */ + +#if defined (__alpha) && W_TYPE_SIZE == 64 +/* There is a bug in g++ before version 5 that +   errors on __builtin_alpha_umulh.  */ +#if !defined(__cplusplus) || __GNUC__ >= 5 +#define umul_ppmm(ph, pl, m0, m1) \ +  do {									\ +    UDItype __m0 = (m0), __m1 = (m1);					\ +    (ph) = __builtin_alpha_umulh (__m0, __m1);				\ +    (pl) = __m0 * __m1;							\ +  } while (0) +#define UMUL_TIME 46 +#endif /* !c++ */ +#ifndef LONGLONG_STANDALONE +#define udiv_qrnnd(q, r, n1, n0, d) \ +  do { UDItype __r;							\ +    (q) = __udiv_qrnnd (&__r, (n1), (n0), (d));				\ +    (r) = __r;								\ +  } while (0) +extern UDItype __udiv_qrnnd (UDItype *, UDItype, UDItype, UDItype); +#define UDIV_TIME 220 +#endif /* LONGLONG_STANDALONE */ +#ifdef __alpha_cix__ +#define count_leading_zeros(COUNT,X)	((COUNT) = __builtin_clzl (X)) +#define count_trailing_zeros(COUNT,X)	((COUNT) = __builtin_ctzl (X)) +#define COUNT_LEADING_ZEROS_0 64 +#else +#define count_leading_zeros(COUNT,X) \ +  do {									\ +    UDItype __xr = (X), __t, __a;					\ +    __t = __builtin_alpha_cmpbge (0, __xr);				\ +    __a = __clz_tab[__t ^ 0xff] - 1;					\ +    __t = __builtin_alpha_extbl (__xr, __a);				\ +    (COUNT) = 64 - (__clz_tab[__t] + __a*8);				\ +  } while (0) +#define count_trailing_zeros(COUNT,X) \ +  do {									\ +    UDItype __xr = (X), __t, __a;					\ +    __t = __builtin_alpha_cmpbge (0, __xr);				\ +    __t = ~__t & -~__t;							\ +    __a = ((__t & 0xCC) != 0) * 2;					\ +    __a += ((__t & 0xF0) != 0) * 4;					\ +    __a += ((__t & 0xAA) != 0);						\ +    __t = __builtin_alpha_extbl (__xr, __a);				\ +    __a <<= 3;								\ +    __t &= -__t;							\ +    __a += ((__t & 0xCC) != 0) * 2;					\ +    __a += ((__t & 0xF0) != 0) * 4;					\ +    __a += ((__t & 0xAA) != 0);						\ +    (COUNT) = __a;							\ +  } while (0) +#endif /* __alpha_cix__ */ +#endif /* __alpha */ + +#if defined (__arc__) && W_TYPE_SIZE == 32 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  __asm__ ("add.f	%1, %4, %5\n\tadc	%0, %2, %3"		\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "%r" ((USItype) (ah)),					\ +	     "rIJ" ((USItype) (bh)),					\ +	     "%r" ((USItype) (al)),					\ +	     "rIJ" ((USItype) (bl))) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  __asm__ ("sub.f	%1, %4, %5\n\tsbc	%0, %2, %3"		\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "r" ((USItype) (ah)),					\ +	     "rIJ" ((USItype) (bh)),					\ +	     "r" ((USItype) (al)),					\ +	     "rIJ" ((USItype) (bl))) + +#define __umulsidi3(u,v) ((UDItype)(USItype)u*(USItype)v) +#ifdef __ARC_NORM__ +#define count_leading_zeros(count, x) \ +  do									\ +    {									\ +      SItype c_;							\ +									\ +      __asm__ ("norm.f\t%0,%1\n\tmov.mi\t%0,-1" : "=r" (c_) : "r" (x) : "cc");\ +      (count) = c_ + 1;							\ +    }									\ +  while (0) +#define COUNT_LEADING_ZEROS_0 32 +#endif +#endif + +#if defined (__arm__) && (defined (__thumb2__) || !defined (__thumb__)) \ + && W_TYPE_SIZE == 32 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  __asm__ ("adds	%1, %4, %5\n\tadc	%0, %2, %3"		\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "%r" ((USItype) (ah)),					\ +	     "rI" ((USItype) (bh)),					\ +	     "%r" ((USItype) (al)),					\ +	     "rI" ((USItype) (bl)) __CLOBBER_CC) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  __asm__ ("subs	%1, %4, %5\n\tsbc	%0, %2, %3"		\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "r" ((USItype) (ah)),					\ +	     "rI" ((USItype) (bh)),					\ +	     "r" ((USItype) (al)),					\ +	     "rI" ((USItype) (bl)) __CLOBBER_CC) +# if defined(__ARM_ARCH_2__) || defined(__ARM_ARCH_2A__) \ +     || defined(__ARM_ARCH_3__) +#  define umul_ppmm(xh, xl, a, b)					\ +  do {									\ +    register USItype __t0, __t1, __t2;					\ +    __asm__ ("%@ Inlined umul_ppmm\n"					\ +	   "	mov	%2, %5, lsr #16\n"				\ +	   "	mov	%0, %6, lsr #16\n"				\ +	   "	bic	%3, %5, %2, lsl #16\n"				\ +	   "	bic	%4, %6, %0, lsl #16\n"				\ +	   "	mul	%1, %3, %4\n"					\ +	   "	mul	%4, %2, %4\n"					\ +	   "	mul	%3, %0, %3\n"					\ +	   "	mul	%0, %2, %0\n"					\ +	   "	adds	%3, %4, %3\n"					\ +	   "	addcs	%0, %0, #65536\n"				\ +	   "	adds	%1, %1, %3, lsl #16\n"				\ +	   "	adc	%0, %0, %3, lsr #16"				\ +	   : "=&r" ((USItype) (xh)),					\ +	     "=r" ((USItype) (xl)),					\ +	     "=&r" (__t0), "=&r" (__t1), "=r" (__t2)			\ +	   : "r" ((USItype) (a)),					\ +	     "r" ((USItype) (b)) __CLOBBER_CC );			\ +  } while (0) +#  define UMUL_TIME 20 +# else +#  define umul_ppmm(xh, xl, a, b)					\ +  do {									\ +    /* Generate umull, under compiler control.  */			\ +    register UDItype __t0 = (UDItype)(USItype)(a) * (USItype)(b);	\ +    (xl) = (USItype)__t0;						\ +    (xh) = (USItype)(__t0 >> 32);					\ +  } while (0) +#  define UMUL_TIME 3 +# endif +# define UDIV_TIME 100 +#endif /* __arm__ */ + +#if defined(__arm__) +/* Let gcc decide how best to implement count_leading_zeros.  */ +#define count_leading_zeros(COUNT,X)	((COUNT) = __builtin_clz (X)) +#define count_trailing_zeros(COUNT,X)   ((COUNT) = __builtin_ctz (X)) +#define COUNT_LEADING_ZEROS_0 32 +#endif + +#if defined (__AVR__) + +#if W_TYPE_SIZE == 16 +#define count_leading_zeros(COUNT,X)  ((COUNT) = __builtin_clz (X)) +#define count_trailing_zeros(COUNT,X) ((COUNT) = __builtin_ctz (X)) +#define COUNT_LEADING_ZEROS_0 16 +#endif /* W_TYPE_SIZE == 16 */ + +#if W_TYPE_SIZE == 32 +#define count_leading_zeros(COUNT,X)  ((COUNT) = __builtin_clzl (X)) +#define count_trailing_zeros(COUNT,X) ((COUNT) = __builtin_ctzl (X)) +#define COUNT_LEADING_ZEROS_0 32 +#endif /* W_TYPE_SIZE == 32 */ + +#if W_TYPE_SIZE == 64 +#define count_leading_zeros(COUNT,X)  ((COUNT) = __builtin_clzll (X)) +#define count_trailing_zeros(COUNT,X) ((COUNT) = __builtin_ctzll (X)) +#define COUNT_LEADING_ZEROS_0 64 +#endif /* W_TYPE_SIZE == 64 */ + +#endif /* defined (__AVR__) */ + +#if defined (__CRIS__) + +#if __CRIS_arch_version >= 3 +#define count_leading_zeros(COUNT, X) ((COUNT) = __builtin_clz (X)) +#define COUNT_LEADING_ZEROS_0 32 +#endif /* __CRIS_arch_version >= 3 */ + +#if __CRIS_arch_version >= 8 +#define count_trailing_zeros(COUNT, X) ((COUNT) = __builtin_ctz (X)) +#endif /* __CRIS_arch_version >= 8 */ + +#if __CRIS_arch_version >= 10 +#define __umulsidi3(u,v) ((UDItype)(USItype) (u) * (UDItype)(USItype) (v)) +#else +#define __umulsidi3 __umulsidi3 +extern UDItype __umulsidi3 (USItype, USItype); +#endif /* __CRIS_arch_version >= 10 */ + +#define umul_ppmm(w1, w0, u, v)		\ +  do {					\ +    UDItype __x = __umulsidi3 (u, v);	\ +    (w0) = (USItype) (__x);		\ +    (w1) = (USItype) (__x >> 32);	\ +  } while (0) + +/* FIXME: defining add_ssaaaa and sub_ddmmss should be advantageous for +   DFmode ("double" intrinsics, avoiding two of the three insns handling +   carry), but defining them as open-code C composing and doing the +   operation in DImode (UDImode) shows that the DImode needs work: +   register pressure from requiring neighboring registers and the +   traffic to and from them come to dominate, in the 4.7 series.  */ + +#endif /* defined (__CRIS__) */ + +#if defined (__hppa) && W_TYPE_SIZE == 32 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  __asm__ ("add %4,%5,%1\n\taddc %2,%3,%0"				\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "%rM" ((USItype) (ah)),					\ +	     "rM" ((USItype) (bh)),					\ +	     "%rM" ((USItype) (al)),					\ +	     "rM" ((USItype) (bl))) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  __asm__ ("sub %4,%5,%1\n\tsubb %2,%3,%0"				\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "rM" ((USItype) (ah)),					\ +	     "rM" ((USItype) (bh)),					\ +	     "rM" ((USItype) (al)),					\ +	     "rM" ((USItype) (bl))) +#if defined (_PA_RISC1_1) +#define umul_ppmm(w1, w0, u, v) \ +  do {									\ +    union								\ +      {									\ +	UDItype __f;							\ +	struct {USItype __w1, __w0;} __w1w0;				\ +      } __t;								\ +    __asm__ ("xmpyu %1,%2,%0"						\ +	     : "=x" (__t.__f)						\ +	     : "x" ((USItype) (u)),					\ +	       "x" ((USItype) (v)));					\ +    (w1) = __t.__w1w0.__w1;						\ +    (w0) = __t.__w1w0.__w0;						\ +     } while (0) +#define UMUL_TIME 8 +#else +#define UMUL_TIME 30 +#endif +#define UDIV_TIME 40 +#define count_leading_zeros(count, x) \ +  do {									\ +    USItype __tmp;							\ +    __asm__ (								\ +       "ldi		1,%0\n"						\ +"	extru,=		%1,15,16,%%r0		; Bits 31..16 zero?\n"	\ +"	extru,tr	%1,15,16,%1		; No.  Shift down, skip add.\n"\ +"	ldo		16(%0),%0		; Yes.  Perform add.\n"	\ +"	extru,=		%1,23,8,%%r0		; Bits 15..8 zero?\n"	\ +"	extru,tr	%1,23,8,%1		; No.  Shift down, skip add.\n"\ +"	ldo		8(%0),%0		; Yes.  Perform add.\n"	\ +"	extru,=		%1,27,4,%%r0		; Bits 7..4 zero?\n"	\ +"	extru,tr	%1,27,4,%1		; No.  Shift down, skip add.\n"\ +"	ldo		4(%0),%0		; Yes.  Perform add.\n"	\ +"	extru,=		%1,29,2,%%r0		; Bits 3..2 zero?\n"	\ +"	extru,tr	%1,29,2,%1		; No.  Shift down, skip add.\n"\ +"	ldo		2(%0),%0		; Yes.  Perform add.\n"	\ +"	extru		%1,30,1,%1		; Extract bit 1.\n"	\ +"	sub		%0,%1,%0		; Subtract it.\n"	\ +	: "=r" (count), "=r" (__tmp) : "1" (x));			\ +  } while (0) +#endif + +#if (defined (__i370__) || defined (__s390__) || defined (__mvs__)) && W_TYPE_SIZE == 32 +#if !defined (__zarch__) +#define smul_ppmm(xh, xl, m0, m1) \ +  do {									\ +    union {DItype __ll;							\ +	   struct {USItype __h, __l;} __i;				\ +	  } __x;							\ +    __asm__ ("lr %N0,%1\n\tmr %0,%2"					\ +	     : "=&r" (__x.__ll)						\ +	     : "r" (m0), "r" (m1));					\ +    (xh) = __x.__i.__h; (xl) = __x.__i.__l;				\ +  } while (0) +#define sdiv_qrnnd(q, r, n1, n0, d) \ +  do {									\ +    union {DItype __ll;							\ +	   struct {USItype __h, __l;} __i;				\ +	  } __x;							\ +    __x.__i.__h = n1; __x.__i.__l = n0;					\ +    __asm__ ("dr %0,%2"							\ +	     : "=r" (__x.__ll)						\ +	     : "0" (__x.__ll), "r" (d));				\ +    (q) = __x.__i.__l; (r) = __x.__i.__h;				\ +  } while (0) +#else +#define smul_ppmm(xh, xl, m0, m1) \ +  do {                                                                  \ +    register SItype __r0 __asm__ ("0");					\ +    register SItype __r1 __asm__ ("1") = (m0);				\ +									\ +    __asm__ ("mr\t%%r0,%3"                                              \ +	     : "=r" (__r0), "=r" (__r1)					\ +	     : "r"  (__r1),  "r" (m1));					\ +    (xh) = __r0; (xl) = __r1;						\ +  } while (0) + +#define sdiv_qrnnd(q, r, n1, n0, d) \ +  do {									\ +    register SItype __r0 __asm__ ("0") = (n1);				\ +    register SItype __r1 __asm__ ("1") = (n0);				\ +									\ +    __asm__ ("dr\t%%r0,%4"                                              \ +	     : "=r" (__r0), "=r" (__r1)					\ +	     : "r" (__r0), "r" (__r1), "r" (d));			\ +    (q) = __r1; (r) = __r0;						\ +  } while (0) +#endif /* __zarch__ */ +#endif + +#if (defined (__i386__) || defined (__i486__)) && W_TYPE_SIZE == 32 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  __asm__ ("add{l} {%5,%1|%1,%5}\n\tadc{l} {%3,%0|%0,%3}"		\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "%0" ((USItype) (ah)),					\ +	     "g" ((USItype) (bh)),					\ +	     "%1" ((USItype) (al)),					\ +	     "g" ((USItype) (bl))) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  __asm__ ("sub{l} {%5,%1|%1,%5}\n\tsbb{l} {%3,%0|%0,%3}"		\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "0" ((USItype) (ah)),					\ +	     "g" ((USItype) (bh)),					\ +	     "1" ((USItype) (al)),					\ +	     "g" ((USItype) (bl))) +#define umul_ppmm(w1, w0, u, v) \ +  __asm__ ("mul{l} %3"							\ +	   : "=a" ((USItype) (w0)),					\ +	     "=d" ((USItype) (w1))					\ +	   : "%0" ((USItype) (u)),					\ +	     "rm" ((USItype) (v))) +#define udiv_qrnnd(q, r, n1, n0, dv) \ +  __asm__ ("div{l} %4"							\ +	   : "=a" ((USItype) (q)),					\ +	     "=d" ((USItype) (r))					\ +	   : "0" ((USItype) (n0)),					\ +	     "1" ((USItype) (n1)),					\ +	     "rm" ((USItype) (dv))) +#define count_leading_zeros(count, x)	((count) = __builtin_clz (x)) +#define count_trailing_zeros(count, x)	((count) = __builtin_ctz (x)) +#define UMUL_TIME 40 +#define UDIV_TIME 40 +#endif /* 80x86 */ + +#if defined (__x86_64__) && W_TYPE_SIZE == 64 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  __asm__ ("add{q} {%5,%1|%1,%5}\n\tadc{q} {%3,%0|%0,%3}"		\ +	   : "=r" ((UDItype) (sh)),					\ +	     "=&r" ((UDItype) (sl))					\ +	   : "%0" ((UDItype) (ah)),					\ +	     "rme" ((UDItype) (bh)),					\ +	     "%1" ((UDItype) (al)),					\ +	     "rme" ((UDItype) (bl))) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  __asm__ ("sub{q} {%5,%1|%1,%5}\n\tsbb{q} {%3,%0|%0,%3}"		\ +	   : "=r" ((UDItype) (sh)),					\ +	     "=&r" ((UDItype) (sl))					\ +	   : "0" ((UDItype) (ah)),					\ +	     "rme" ((UDItype) (bh)),					\ +	     "1" ((UDItype) (al)),					\ +	     "rme" ((UDItype) (bl))) +#define umul_ppmm(w1, w0, u, v) \ +  __asm__ ("mul{q} %3"							\ +	   : "=a" ((UDItype) (w0)),					\ +	     "=d" ((UDItype) (w1))					\ +	   : "%0" ((UDItype) (u)),					\ +	     "rm" ((UDItype) (v))) +#define udiv_qrnnd(q, r, n1, n0, dv) \ +  __asm__ ("div{q} %4"							\ +	   : "=a" ((UDItype) (q)),					\ +	     "=d" ((UDItype) (r))					\ +	   : "0" ((UDItype) (n0)),					\ +	     "1" ((UDItype) (n1)),					\ +	     "rm" ((UDItype) (dv))) +#define count_leading_zeros(count, x)	((count) = __builtin_clzll (x)) +#define count_trailing_zeros(count, x)	((count) = __builtin_ctzll (x)) +#define UMUL_TIME 40 +#define UDIV_TIME 40 +#endif /* x86_64 */ + +#if defined (__i960__) && W_TYPE_SIZE == 32 +#define umul_ppmm(w1, w0, u, v) \ +  ({union {UDItype __ll;						\ +	   struct {USItype __l, __h;} __i;				\ +	  } __xx;							\ +  __asm__ ("emul	%2,%1,%0"					\ +	   : "=d" (__xx.__ll)						\ +	   : "%dI" ((USItype) (u)),					\ +	     "dI" ((USItype) (v)));					\ +  (w1) = __xx.__i.__h; (w0) = __xx.__i.__l;}) +#define __umulsidi3(u, v) \ +  ({UDItype __w;							\ +    __asm__ ("emul	%2,%1,%0"					\ +	     : "=d" (__w)						\ +	     : "%dI" ((USItype) (u)),					\ +	       "dI" ((USItype) (v)));					\ +    __w; }) +#endif /* __i960__ */ + +#if defined (__ia64) && W_TYPE_SIZE == 64 +/* This form encourages gcc (pre-release 3.4 at least) to emit predicated +   "sub r=r,r" and "sub r=r,r,1", giving a 2 cycle latency.  The generic +   code using "al<bl" arithmetically comes out making an actual 0 or 1 in a +   register, which takes an extra cycle.  */ +#define sub_ddmmss(sh, sl, ah, al, bh, bl)				\ +  do {									\ +    UWtype __x;								\ +    __x = (al) - (bl);							\ +    if ((al) < (bl))							\ +      (sh) = (ah) - (bh) - 1;						\ +    else								\ +      (sh) = (ah) - (bh);						\ +    (sl) = __x;								\ +  } while (0) + +/* Do both product parts in assembly, since that gives better code with +   all gcc versions.  Some callers will just use the upper part, and in +   that situation we waste an instruction, but not any cycles.  */ +#define umul_ppmm(ph, pl, m0, m1)					\ +  __asm__ ("xma.hu %0 = %2, %3, f0\n\txma.l %1 = %2, %3, f0"		\ +	   : "=&f" (ph), "=f" (pl)					\ +	   : "f" (m0), "f" (m1)) +#define count_leading_zeros(count, x)					\ +  do {									\ +    UWtype _x = (x), _y, _a, _c;					\ +    __asm__ ("mux1 %0 = %1, @rev" : "=r" (_y) : "r" (_x));		\ +    __asm__ ("czx1.l %0 = %1" : "=r" (_a) : "r" (-_y | _y));		\ +    _c = (_a - 1) << 3;							\ +    _x >>= _c;								\ +    if (_x >= 1 << 4)							\ +      _x >>= 4, _c += 4;						\ +    if (_x >= 1 << 2)							\ +      _x >>= 2, _c += 2;						\ +    _c += _x >> 1;							\ +    (count) =  W_TYPE_SIZE - 1 - _c;					\ +  } while (0) +/* similar to what gcc does for __builtin_ffs, but 0 based rather than 1 +   based, and we don't need a special case for x==0 here */ +#define count_trailing_zeros(count, x)					\ +  do {									\ +    UWtype __ctz_x = (x);						\ +    __asm__ ("popcnt %0 = %1"						\ +	     : "=r" (count)						\ +	     : "r" ((__ctz_x-1) & ~__ctz_x));				\ +  } while (0) +#define UMUL_TIME 14 +#endif + +#if defined (__M32R__) && W_TYPE_SIZE == 32 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  /* The cmp clears the condition bit.  */ \ +  __asm__ ("cmp %0,%0\n\taddx %1,%5\n\taddx %0,%3"			\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "0" ((USItype) (ah)),					\ +	     "r" ((USItype) (bh)),					\ +	     "1" ((USItype) (al)),					\ +	     "r" ((USItype) (bl))					\ +	   : "cbit") +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  /* The cmp clears the condition bit.  */ \ +  __asm__ ("cmp %0,%0\n\tsubx %1,%5\n\tsubx %0,%3"			\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "0" ((USItype) (ah)),					\ +	     "r" ((USItype) (bh)),					\ +	     "1" ((USItype) (al)),					\ +	     "r" ((USItype) (bl))					\ +	   : "cbit") +#endif /* __M32R__ */ + +#if defined (__mc68000__) && W_TYPE_SIZE == 32 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  __asm__ ("add%.l %5,%1\n\taddx%.l %3,%0"				\ +	   : "=d" ((USItype) (sh)),					\ +	     "=&d" ((USItype) (sl))					\ +	   : "%0" ((USItype) (ah)),					\ +	     "d" ((USItype) (bh)),					\ +	     "%1" ((USItype) (al)),					\ +	     "g" ((USItype) (bl))) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  __asm__ ("sub%.l %5,%1\n\tsubx%.l %3,%0"				\ +	   : "=d" ((USItype) (sh)),					\ +	     "=&d" ((USItype) (sl))					\ +	   : "0" ((USItype) (ah)),					\ +	     "d" ((USItype) (bh)),					\ +	     "1" ((USItype) (al)),					\ +	     "g" ((USItype) (bl))) + +/* The '020, '030, '040, '060 and CPU32 have 32x32->64 and 64/32->32q-32r.  */ +#if (defined (__mc68020__) && !defined (__mc68060__)) +#define umul_ppmm(w1, w0, u, v) \ +  __asm__ ("mulu%.l %3,%1:%0"						\ +	   : "=d" ((USItype) (w0)),					\ +	     "=d" ((USItype) (w1))					\ +	   : "%0" ((USItype) (u)),					\ +	     "dmi" ((USItype) (v))) +#define UMUL_TIME 45 +#define udiv_qrnnd(q, r, n1, n0, d) \ +  __asm__ ("divu%.l %4,%1:%0"						\ +	   : "=d" ((USItype) (q)),					\ +	     "=d" ((USItype) (r))					\ +	   : "0" ((USItype) (n0)),					\ +	     "1" ((USItype) (n1)),					\ +	     "dmi" ((USItype) (d))) +#define UDIV_TIME 90 +#define sdiv_qrnnd(q, r, n1, n0, d) \ +  __asm__ ("divs%.l %4,%1:%0"						\ +	   : "=d" ((USItype) (q)),					\ +	     "=d" ((USItype) (r))					\ +	   : "0" ((USItype) (n0)),					\ +	     "1" ((USItype) (n1)),					\ +	     "dmi" ((USItype) (d))) + +#elif defined (__mcoldfire__) /* not mc68020 */ + +#define umul_ppmm(xh, xl, a, b) \ +  __asm__ ("| Inlined umul_ppmm\n"					\ +	   "	move%.l	%2,%/d0\n"					\ +	   "	move%.l	%3,%/d1\n"					\ +	   "	move%.l	%/d0,%/d2\n"					\ +	   "	swap	%/d0\n"						\ +	   "	move%.l	%/d1,%/d3\n"					\ +	   "	swap	%/d1\n"						\ +	   "	move%.w	%/d2,%/d4\n"					\ +	   "	mulu	%/d3,%/d4\n"					\ +	   "	mulu	%/d1,%/d2\n"					\ +	   "	mulu	%/d0,%/d3\n"					\ +	   "	mulu	%/d0,%/d1\n"					\ +	   "	move%.l	%/d4,%/d0\n"					\ +	   "	clr%.w	%/d0\n"						\ +	   "	swap	%/d0\n"						\ +	   "	add%.l	%/d0,%/d2\n"					\ +	   "	add%.l	%/d3,%/d2\n"					\ +	   "	jcc	1f\n"						\ +	   "	add%.l	%#65536,%/d1\n"					\ +	   "1:	swap	%/d2\n"						\ +	   "	moveq	%#0,%/d0\n"					\ +	   "	move%.w	%/d2,%/d0\n"					\ +	   "	move%.w	%/d4,%/d2\n"					\ +	   "	move%.l	%/d2,%1\n"					\ +	   "	add%.l	%/d1,%/d0\n"					\ +	   "	move%.l	%/d0,%0"					\ +	   : "=g" ((USItype) (xh)),					\ +	     "=g" ((USItype) (xl))					\ +	   : "g" ((USItype) (a)),					\ +	     "g" ((USItype) (b))					\ +	   : "d0", "d1", "d2", "d3", "d4") +#define UMUL_TIME 100 +#define UDIV_TIME 400 +#else /* not ColdFire */ +/* %/ inserts REGISTER_PREFIX, %# inserts IMMEDIATE_PREFIX.  */ +#define umul_ppmm(xh, xl, a, b) \ +  __asm__ ("| Inlined umul_ppmm\n"					\ +	   "	move%.l	%2,%/d0\n"					\ +	   "	move%.l	%3,%/d1\n"					\ +	   "	move%.l	%/d0,%/d2\n"					\ +	   "	swap	%/d0\n"						\ +	   "	move%.l	%/d1,%/d3\n"					\ +	   "	swap	%/d1\n"						\ +	   "	move%.w	%/d2,%/d4\n"					\ +	   "	mulu	%/d3,%/d4\n"					\ +	   "	mulu	%/d1,%/d2\n"					\ +	   "	mulu	%/d0,%/d3\n"					\ +	   "	mulu	%/d0,%/d1\n"					\ +	   "	move%.l	%/d4,%/d0\n"					\ +	   "	eor%.w	%/d0,%/d0\n"					\ +	   "	swap	%/d0\n"						\ +	   "	add%.l	%/d0,%/d2\n"					\ +	   "	add%.l	%/d3,%/d2\n"					\ +	   "	jcc	1f\n"						\ +	   "	add%.l	%#65536,%/d1\n"					\ +	   "1:	swap	%/d2\n"						\ +	   "	moveq	%#0,%/d0\n"					\ +	   "	move%.w	%/d2,%/d0\n"					\ +	   "	move%.w	%/d4,%/d2\n"					\ +	   "	move%.l	%/d2,%1\n"					\ +	   "	add%.l	%/d1,%/d0\n"					\ +	   "	move%.l	%/d0,%0"					\ +	   : "=g" ((USItype) (xh)),					\ +	     "=g" ((USItype) (xl))					\ +	   : "g" ((USItype) (a)),					\ +	     "g" ((USItype) (b))					\ +	   : "d0", "d1", "d2", "d3", "d4") +#define UMUL_TIME 100 +#define UDIV_TIME 400 + +#endif /* not mc68020 */ + +/* The '020, '030, '040 and '060 have bitfield insns. +   cpu32 disguises as a 68020, but lacks them.  */ +#if defined (__mc68020__) && !defined (__mcpu32__) +#define count_leading_zeros(count, x) \ +  __asm__ ("bfffo %1{%b2:%b2},%0"					\ +	   : "=d" ((USItype) (count))					\ +	   : "od" ((USItype) (x)), "n" (0)) +/* Some ColdFire architectures have a ff1 instruction supported via +   __builtin_clz. */ +#elif defined (__mcfisaaplus__) || defined (__mcfisac__) +#define count_leading_zeros(count,x) ((count) = __builtin_clz (x)) +#define COUNT_LEADING_ZEROS_0 32 +#endif +#endif /* mc68000 */ + +#if defined (__m88000__) && W_TYPE_SIZE == 32 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  __asm__ ("addu.co %1,%r4,%r5\n\taddu.ci %0,%r2,%r3"			\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "%rJ" ((USItype) (ah)),					\ +	     "rJ" ((USItype) (bh)),					\ +	     "%rJ" ((USItype) (al)),					\ +	     "rJ" ((USItype) (bl))) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  __asm__ ("subu.co %1,%r4,%r5\n\tsubu.ci %0,%r2,%r3"			\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "rJ" ((USItype) (ah)),					\ +	     "rJ" ((USItype) (bh)),					\ +	     "rJ" ((USItype) (al)),					\ +	     "rJ" ((USItype) (bl))) +#define count_leading_zeros(count, x) \ +  do {									\ +    USItype __cbtmp;							\ +    __asm__ ("ff1 %0,%1"						\ +	     : "=r" (__cbtmp)						\ +	     : "r" ((USItype) (x)));					\ +    (count) = __cbtmp ^ 31;						\ +  } while (0) +#define COUNT_LEADING_ZEROS_0 63 /* sic */ +#if defined (__mc88110__) +#define umul_ppmm(wh, wl, u, v) \ +  do {									\ +    union {UDItype __ll;						\ +	   struct {USItype __h, __l;} __i;				\ +	  } __xx;							\ +    __asm__ ("mulu.d	%0,%1,%2"					\ +	     : "=r" (__xx.__ll)						\ +	     : "r" ((USItype) (u)),					\ +	       "r" ((USItype) (v)));					\ +    (wh) = __xx.__i.__h;						\ +    (wl) = __xx.__i.__l;						\ +  } while (0) +#define udiv_qrnnd(q, r, n1, n0, d) \ +  ({union {UDItype __ll;						\ +	   struct {USItype __h, __l;} __i;				\ +	  } __xx;							\ +  USItype __q;								\ +  __xx.__i.__h = (n1); __xx.__i.__l = (n0);				\ +  __asm__ ("divu.d %0,%1,%2"						\ +	   : "=r" (__q)							\ +	   : "r" (__xx.__ll),						\ +	     "r" ((USItype) (d)));					\ +  (r) = (n0) - __q * (d); (q) = __q; }) +#define UMUL_TIME 5 +#define UDIV_TIME 25 +#else +#define UMUL_TIME 17 +#define UDIV_TIME 150 +#endif /* __mc88110__ */ +#endif /* __m88000__ */ + +#if defined (__mn10300__) +# if defined (__AM33__) +#  define count_leading_zeros(COUNT,X)	((COUNT) = __builtin_clz (X)) +#  define umul_ppmm(w1, w0, u, v)		\ +    asm("mulu %3,%2,%1,%0" : "=r"(w0), "=r"(w1) : "r"(u), "r"(v)) +#  define smul_ppmm(w1, w0, u, v)		\ +    asm("mul %3,%2,%1,%0" : "=r"(w0), "=r"(w1) : "r"(u), "r"(v)) +# else +#  define umul_ppmm(w1, w0, u, v)		\ +    asm("nop; nop; mulu %3,%0" : "=d"(w0), "=z"(w1) : "%0"(u), "d"(v)) +#  define smul_ppmm(w1, w0, u, v)		\ +    asm("nop; nop; mul %3,%0" : "=d"(w0), "=z"(w1) : "%0"(u), "d"(v)) +# endif +# define add_ssaaaa(sh, sl, ah, al, bh, bl)	\ +  do {						\ +    DWunion __s, __a, __b;			\ +    __a.s.low = (al); __a.s.high = (ah);	\ +    __b.s.low = (bl); __b.s.high = (bh);	\ +    __s.ll = __a.ll + __b.ll;			\ +    (sl) = __s.s.low; (sh) = __s.s.high;	\ +  } while (0) +# define sub_ddmmss(sh, sl, ah, al, bh, bl)	\ +  do {						\ +    DWunion __s, __a, __b;			\ +    __a.s.low = (al); __a.s.high = (ah);	\ +    __b.s.low = (bl); __b.s.high = (bh);	\ +    __s.ll = __a.ll - __b.ll;			\ +    (sl) = __s.s.low; (sh) = __s.s.high;	\ +  } while (0) +# define udiv_qrnnd(q, r, nh, nl, d)		\ +  asm("divu %2,%0" : "=D"(q), "=z"(r) : "D"(d), "0"(nl), "1"(nh)) +# define sdiv_qrnnd(q, r, nh, nl, d)		\ +  asm("div %2,%0" : "=D"(q), "=z"(r) : "D"(d), "0"(nl), "1"(nh)) +# define UMUL_TIME 3 +# define UDIV_TIME 38 +#endif + +#if defined (__mips__) && W_TYPE_SIZE == 32 +#define umul_ppmm(w1, w0, u, v)						\ +  do {									\ +    UDItype __x = (UDItype) (USItype) (u) * (USItype) (v);		\ +    (w1) = (USItype) (__x >> 32);					\ +    (w0) = (USItype) (__x);						\ +  } while (0) +#define UMUL_TIME 10 +#define UDIV_TIME 100 + +#if (__mips == 32 || __mips == 64) && ! defined (__mips16) +#define count_leading_zeros(COUNT,X)	((COUNT) = __builtin_clz (X)) +#define COUNT_LEADING_ZEROS_0 32 +#endif +#endif /* __mips__ */ + +#if defined (__ns32000__) && W_TYPE_SIZE == 32 +#define umul_ppmm(w1, w0, u, v) \ +  ({union {UDItype __ll;						\ +	   struct {USItype __l, __h;} __i;				\ +	  } __xx;							\ +  __asm__ ("meid %2,%0"							\ +	   : "=g" (__xx.__ll)						\ +	   : "%0" ((USItype) (u)),					\ +	     "g" ((USItype) (v)));					\ +  (w1) = __xx.__i.__h; (w0) = __xx.__i.__l;}) +#define __umulsidi3(u, v) \ +  ({UDItype __w;							\ +    __asm__ ("meid %2,%0"						\ +	     : "=g" (__w)						\ +	     : "%0" ((USItype) (u)),					\ +	       "g" ((USItype) (v)));					\ +    __w; }) +#define udiv_qrnnd(q, r, n1, n0, d) \ +  ({union {UDItype __ll;						\ +	   struct {USItype __l, __h;} __i;				\ +	  } __xx;							\ +  __xx.__i.__h = (n1); __xx.__i.__l = (n0);				\ +  __asm__ ("deid %2,%0"							\ +	   : "=g" (__xx.__ll)						\ +	   : "0" (__xx.__ll),						\ +	     "g" ((USItype) (d)));					\ +  (r) = __xx.__i.__l; (q) = __xx.__i.__h; }) +#define count_trailing_zeros(count,x) \ +  do {									\ +    __asm__ ("ffsd     %2,%0"						\ +	    : "=r" ((USItype) (count))					\ +	    : "0" ((USItype) 0),					\ +	      "r" ((USItype) (x)));					\ +  } while (0) +#endif /* __ns32000__ */ + +/* FIXME: We should test _IBMR2 here when we add assembly support for the +   system vendor compilers. +   FIXME: What's needed for gcc PowerPC VxWorks?  __vxworks__ is not good +   enough, since that hits ARM and m68k too.  */ +#if (defined (_ARCH_PPC)	/* AIX */				\ +     || defined (__powerpc__)	/* gcc */				\ +     || defined (__POWERPC__)	/* BEOS */				\ +     || defined (__ppc__)	/* Darwin */				\ +     || (defined (PPC) && ! defined (CPU_FAMILY)) /* gcc 2.7.x GNU&SysV */    \ +     || (defined (PPC) && defined (CPU_FAMILY)    /* VxWorks */               \ +	 && CPU_FAMILY == PPC)                                                \ +     ) && W_TYPE_SIZE == 32 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  do {									\ +    if (__builtin_constant_p (bh) && (bh) == 0)				\ +      __asm__ ("add%I4c %1,%3,%4\n\taddze %0,%2"		\ +	     : "=r" (sh), "=&r" (sl) : "r" (ah), "%r" (al), "rI" (bl));\ +    else if (__builtin_constant_p (bh) && (bh) == ~(USItype) 0)		\ +      __asm__ ("add%I4c %1,%3,%4\n\taddme %0,%2"		\ +	     : "=r" (sh), "=&r" (sl) : "r" (ah), "%r" (al), "rI" (bl));\ +    else								\ +      __asm__ ("add%I5c %1,%4,%5\n\tadde %0,%2,%3"		\ +	     : "=r" (sh), "=&r" (sl)					\ +	     : "%r" (ah), "r" (bh), "%r" (al), "rI" (bl));		\ +  } while (0) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  do {									\ +    if (__builtin_constant_p (ah) && (ah) == 0)				\ +      __asm__ ("subf%I3c %1,%4,%3\n\tsubfze %0,%2"	\ +	       : "=r" (sh), "=&r" (sl) : "r" (bh), "rI" (al), "r" (bl));\ +    else if (__builtin_constant_p (ah) && (ah) == ~(USItype) 0)		\ +      __asm__ ("subf%I3c %1,%4,%3\n\tsubfme %0,%2"	\ +	       : "=r" (sh), "=&r" (sl) : "r" (bh), "rI" (al), "r" (bl));\ +    else if (__builtin_constant_p (bh) && (bh) == 0)			\ +      __asm__ ("subf%I3c %1,%4,%3\n\taddme %0,%2"		\ +	       : "=r" (sh), "=&r" (sl) : "r" (ah), "rI" (al), "r" (bl));\ +    else if (__builtin_constant_p (bh) && (bh) == ~(USItype) 0)		\ +      __asm__ ("subf%I3c %1,%4,%3\n\taddze %0,%2"		\ +	       : "=r" (sh), "=&r" (sl) : "r" (ah), "rI" (al), "r" (bl));\ +    else								\ +      __asm__ ("subf%I4c %1,%5,%4\n\tsubfe %0,%3,%2"	\ +	       : "=r" (sh), "=&r" (sl)					\ +	       : "r" (ah), "r" (bh), "rI" (al), "r" (bl));		\ +  } while (0) +#define count_leading_zeros(count, x) \ +  __asm__ ("cntlzw %0,%1" : "=r" (count) : "r" (x)) +#define COUNT_LEADING_ZEROS_0 32 +#if defined (_ARCH_PPC) || defined (__powerpc__) || defined (__POWERPC__) \ +  || defined (__ppc__)                                                    \ +  || (defined (PPC) && ! defined (CPU_FAMILY)) /* gcc 2.7.x GNU&SysV */       \ +  || (defined (PPC) && defined (CPU_FAMILY)    /* VxWorks */                  \ +	 && CPU_FAMILY == PPC) +#define umul_ppmm(ph, pl, m0, m1) \ +  do {									\ +    USItype __m0 = (m0), __m1 = (m1);					\ +    __asm__ ("mulhwu %0,%1,%2" : "=r" (ph) : "%r" (m0), "r" (m1));	\ +    (pl) = __m0 * __m1;							\ +  } while (0) +#define UMUL_TIME 15 +#define smul_ppmm(ph, pl, m0, m1) \ +  do {									\ +    SItype __m0 = (m0), __m1 = (m1);					\ +    __asm__ ("mulhw %0,%1,%2" : "=r" (ph) : "%r" (m0), "r" (m1));	\ +    (pl) = __m0 * __m1;							\ +  } while (0) +#define SMUL_TIME 14 +#define UDIV_TIME 120 +#endif +#endif /* 32-bit POWER architecture variants.  */ + +/* We should test _IBMR2 here when we add assembly support for the system +   vendor compilers.  */ +#if (defined (_ARCH_PPC64) || defined (__powerpc64__)) && W_TYPE_SIZE == 64 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  do {									\ +    if (__builtin_constant_p (bh) && (bh) == 0)				\ +      __asm__ ("add%I4c %1,%3,%4\n\taddze %0,%2"		\ +	     : "=r" (sh), "=&r" (sl) : "r" (ah), "%r" (al), "rI" (bl));\ +    else if (__builtin_constant_p (bh) && (bh) == ~(UDItype) 0)		\ +      __asm__ ("add%I4c %1,%3,%4\n\taddme %0,%2"		\ +	     : "=r" (sh), "=&r" (sl) : "r" (ah), "%r" (al), "rI" (bl));\ +    else								\ +      __asm__ ("add%I5c %1,%4,%5\n\tadde %0,%2,%3"		\ +	     : "=r" (sh), "=&r" (sl)					\ +	     : "%r" (ah), "r" (bh), "%r" (al), "rI" (bl));		\ +  } while (0) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  do {									\ +    if (__builtin_constant_p (ah) && (ah) == 0)				\ +      __asm__ ("subf%I3c %1,%4,%3\n\tsubfze %0,%2"	\ +	       : "=r" (sh), "=&r" (sl) : "r" (bh), "rI" (al), "r" (bl));\ +    else if (__builtin_constant_p (ah) && (ah) == ~(UDItype) 0)		\ +      __asm__ ("subf%I3c %1,%4,%3\n\tsubfme %0,%2"	\ +	       : "=r" (sh), "=&r" (sl) : "r" (bh), "rI" (al), "r" (bl));\ +    else if (__builtin_constant_p (bh) && (bh) == 0)			\ +      __asm__ ("subf%I3c %1,%4,%3\n\taddme %0,%2"		\ +	       : "=r" (sh), "=&r" (sl) : "r" (ah), "rI" (al), "r" (bl));\ +    else if (__builtin_constant_p (bh) && (bh) == ~(UDItype) 0)		\ +      __asm__ ("subf%I3c %1,%4,%3\n\taddze %0,%2"		\ +	       : "=r" (sh), "=&r" (sl) : "r" (ah), "rI" (al), "r" (bl));\ +    else								\ +      __asm__ ("subf%I4c %1,%5,%4\n\tsubfe %0,%3,%2"	\ +	       : "=r" (sh), "=&r" (sl)					\ +	       : "r" (ah), "r" (bh), "rI" (al), "r" (bl));		\ +  } while (0) +#define count_leading_zeros(count, x) \ +  __asm__ ("cntlzd %0,%1" : "=r" (count) : "r" (x)) +#define COUNT_LEADING_ZEROS_0 64 +#define umul_ppmm(ph, pl, m0, m1) \ +  do {									\ +    UDItype __m0 = (m0), __m1 = (m1);					\ +    __asm__ ("mulhdu %0,%1,%2" : "=r" (ph) : "%r" (m0), "r" (m1));	\ +    (pl) = __m0 * __m1;							\ +  } while (0) +#define UMUL_TIME 15 +#define smul_ppmm(ph, pl, m0, m1) \ +  do {									\ +    DItype __m0 = (m0), __m1 = (m1);					\ +    __asm__ ("mulhd %0,%1,%2" : "=r" (ph) : "%r" (m0), "r" (m1));	\ +    (pl) = __m0 * __m1;							\ +  } while (0) +#define SMUL_TIME 14  /* ??? */ +#define UDIV_TIME 120 /* ??? */ +#endif /* 64-bit PowerPC.  */ + +#if defined (__ibm032__) /* RT/ROMP */ && W_TYPE_SIZE == 32 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  __asm__ ("a %1,%5\n\tae %0,%3"					\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "%0" ((USItype) (ah)),					\ +	     "r" ((USItype) (bh)),					\ +	     "%1" ((USItype) (al)),					\ +	     "r" ((USItype) (bl))) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  __asm__ ("s %1,%5\n\tse %0,%3"					\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "0" ((USItype) (ah)),					\ +	     "r" ((USItype) (bh)),					\ +	     "1" ((USItype) (al)),					\ +	     "r" ((USItype) (bl))) +#define umul_ppmm(ph, pl, m0, m1) \ +  do {									\ +    USItype __m0 = (m0), __m1 = (m1);					\ +    __asm__ (								\ +       "s	r2,r2\n"						\ +"	mts	r10,%2\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	m	r2,%3\n"						\ +"	cas	%0,r2,r0\n"						\ +"	mfs	r10,%1"							\ +	     : "=r" ((USItype) (ph)),					\ +	       "=r" ((USItype) (pl))					\ +	     : "%r" (__m0),						\ +		"r" (__m1)						\ +	     : "r2");							\ +    (ph) += ((((SItype) __m0 >> 31) & __m1)				\ +	     + (((SItype) __m1 >> 31) & __m0));				\ +  } while (0) +#define UMUL_TIME 20 +#define UDIV_TIME 200 +#define count_leading_zeros(count, x) \ +  do {									\ +    if ((x) >= 0x10000)							\ +      __asm__ ("clz	%0,%1"						\ +	       : "=r" ((USItype) (count))				\ +	       : "r" ((USItype) (x) >> 16));				\ +    else								\ +      {									\ +	__asm__ ("clz	%0,%1"						\ +		 : "=r" ((USItype) (count))				\ +		 : "r" ((USItype) (x)));					\ +	(count) += 16;							\ +      }									\ +  } while (0) +#endif + +#if defined(__sh__) && (!defined (__SHMEDIA__) || !__SHMEDIA__) && W_TYPE_SIZE == 32 +#ifndef __sh1__ +#define umul_ppmm(w1, w0, u, v) \ +  __asm__ (								\ +       "dmulu.l	%2,%3\n\tsts%M1	macl,%1\n\tsts%M0	mach,%0"	\ +	   : "=r<" ((USItype)(w1)),					\ +	     "=r<" ((USItype)(w0))					\ +	   : "r" ((USItype)(u)),					\ +	     "r" ((USItype)(v))						\ +	   : "macl", "mach") +#define UMUL_TIME 5 +#endif + +/* This is the same algorithm as __udiv_qrnnd_c.  */ +#define UDIV_NEEDS_NORMALIZATION 1 + +#ifdef __FDPIC__ +/* FDPIC needs a special version of the asm fragment to extract the +   code address from the function descriptor. __udiv_qrnnd_16 is +   assumed to be local and not to use the GOT, so loading r12 is +   not needed. */ +#define udiv_qrnnd(q, r, n1, n0, d) \ +  do {									\ +    extern UWtype __udiv_qrnnd_16 (UWtype, UWtype)			\ +			__attribute__ ((visibility ("hidden")));	\ +    /* r0: rn r1: qn */ /* r0: n1 r4: n0 r5: d r6: d1 */ /* r2: __m */	\ +    __asm__ (								\ +	"mov%M4	%4,r5\n"						\ +"	swap.w	%3,r4\n"						\ +"	swap.w	r5,r6\n"						\ +"	mov.l	@%5,r2\n"						\ +"	jsr	@r2\n"							\ +"	shll16	r6\n"							\ +"	swap.w	r4,r4\n"						\ +"	mov.l	@%5,r2\n"						\ +"	jsr	@r2\n"							\ +"	swap.w	r1,%0\n"						\ +"	or	r1,%0"							\ +	: "=r" (q), "=&z" (r)						\ +	: "1" (n1), "r" (n0), "rm" (d), "r" (&__udiv_qrnnd_16)		\ +	: "r1", "r2", "r4", "r5", "r6", "pr", "t");			\ +  } while (0) +#else +#define udiv_qrnnd(q, r, n1, n0, d) \ +  do {									\ +    extern UWtype __udiv_qrnnd_16 (UWtype, UWtype)			\ +			__attribute__ ((visibility ("hidden")));	\ +    /* r0: rn r1: qn */ /* r0: n1 r4: n0 r5: d r6: d1 */ /* r2: __m */	\ +    __asm__ (								\ +	"mov%M4 %4,r5\n"						\ +"	swap.w %3,r4\n"							\ +"	swap.w r5,r6\n"							\ +"	jsr @%5\n"							\ +"	shll16 r6\n"							\ +"	swap.w r4,r4\n"							\ +"	jsr @%5\n"							\ +"	swap.w r1,%0\n"							\ +"	or r1,%0"							\ +	: "=r" (q), "=&z" (r)						\ +	: "1" (n1), "r" (n0), "rm" (d), "r" (&__udiv_qrnnd_16)		\ +	: "r1", "r2", "r4", "r5", "r6", "pr", "t");			\ +  } while (0) +#endif /* __FDPIC__  */ + +#define UDIV_TIME 80 + +#define sub_ddmmss(sh, sl, ah, al, bh, bl)				\ +  __asm__ ("clrt;subc %5,%1; subc %4,%0"				\ +	   : "=r" (sh), "=r" (sl)					\ +	   : "0" (ah), "1" (al), "r" (bh), "r" (bl) : "t") + +#endif /* __sh__ */ + +#if defined (__SH5__) && defined (__SHMEDIA__) && __SHMEDIA__ && W_TYPE_SIZE == 32 +#define __umulsidi3(u,v) ((UDItype)(USItype)u*(USItype)v) +#define count_leading_zeros(count, x) \ +  do									\ +    {									\ +      UDItype x_ = (USItype)(x);					\ +      SItype c_;							\ +									\ +      __asm__ ("nsb %1, %0" : "=r" (c_) : "r" (x_));			\ +      (count) = c_ - 31;						\ +    }									\ +  while (0) +#define COUNT_LEADING_ZEROS_0 32 +#endif + +#if defined (__sparc__) && !defined (__arch64__) && !defined (__sparcv9) \ +    && W_TYPE_SIZE == 32 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  __asm__ ("addcc %r4,%5,%1\n\taddx %r2,%3,%0"				\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "%rJ" ((USItype) (ah)),					\ +	     "rI" ((USItype) (bh)),					\ +	     "%rJ" ((USItype) (al)),					\ +	     "rI" ((USItype) (bl))					\ +	   __CLOBBER_CC) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  __asm__ ("subcc %r4,%5,%1\n\tsubx %r2,%3,%0"				\ +	   : "=r" ((USItype) (sh)),					\ +	     "=&r" ((USItype) (sl))					\ +	   : "rJ" ((USItype) (ah)),					\ +	     "rI" ((USItype) (bh)),					\ +	     "rJ" ((USItype) (al)),					\ +	     "rI" ((USItype) (bl))					\ +	   __CLOBBER_CC) +#if defined (__sparc_v9__) +#define umul_ppmm(w1, w0, u, v) \ +  do {									\ +    register USItype __g1 asm ("g1");					\ +    __asm__ ("umul\t%2,%3,%1\n\t"					\ +	     "srlx\t%1, 32, %0"						\ +	     : "=r" ((USItype) (w1)),					\ +	       "=r" (__g1)						\ +	     : "r" ((USItype) (u)),					\ +	       "r" ((USItype) (v)));					\ +    (w0) = __g1;							\ +  } while (0) +#define udiv_qrnnd(__q, __r, __n1, __n0, __d) \ +  __asm__ ("mov\t%2,%%y\n\t"						\ +	   "udiv\t%3,%4,%0\n\t"						\ +	   "umul\t%0,%4,%1\n\t"						\ +	   "sub\t%3,%1,%1"						\ +	   : "=&r" ((USItype) (__q)),					\ +	     "=&r" ((USItype) (__r))					\ +	   : "r" ((USItype) (__n1)),					\ +	     "r" ((USItype) (__n0)),					\ +	     "r" ((USItype) (__d))) +#else +#if defined (__sparc_v8__) +#define umul_ppmm(w1, w0, u, v) \ +  __asm__ ("umul %2,%3,%1;rd %%y,%0"					\ +	   : "=r" ((USItype) (w1)),					\ +	     "=r" ((USItype) (w0))					\ +	   : "r" ((USItype) (u)),					\ +	     "r" ((USItype) (v))) +#define udiv_qrnnd(__q, __r, __n1, __n0, __d) \ +  __asm__ ("mov %2,%%y;nop;nop;nop;udiv %3,%4,%0;umul %0,%4,%1;sub %3,%1,%1"\ +	   : "=&r" ((USItype) (__q)),					\ +	     "=&r" ((USItype) (__r))					\ +	   : "r" ((USItype) (__n1)),					\ +	     "r" ((USItype) (__n0)),					\ +	     "r" ((USItype) (__d))) +#else +#if defined (__sparclite__) +/* This has hardware multiply but not divide.  It also has two additional +   instructions scan (ffs from high bit) and divscc.  */ +#define umul_ppmm(w1, w0, u, v) \ +  __asm__ ("umul %2,%3,%1;rd %%y,%0"					\ +	   : "=r" ((USItype) (w1)),					\ +	     "=r" ((USItype) (w0))					\ +	   : "r" ((USItype) (u)),					\ +	     "r" ((USItype) (v))) +#define udiv_qrnnd(q, r, n1, n0, d) \ +  __asm__ ("! Inlined udiv_qrnnd\n"					\ +"	wr	%%g0,%2,%%y	! Not a delayed write for sparclite\n"	\ +"	tst	%%g0\n"							\ +"	divscc	%3,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%%g1\n"						\ +"	divscc	%%g1,%4,%0\n"						\ +"	rd	%%y,%1\n"						\ +"	bl,a 1f\n"							\ +"	add	%1,%4,%1\n"						\ +"1:	! End of inline udiv_qrnnd"					\ +	   : "=r" ((USItype) (q)),					\ +	     "=r" ((USItype) (r))					\ +	   : "r" ((USItype) (n1)),					\ +	     "r" ((USItype) (n0)),					\ +	     "rI" ((USItype) (d))					\ +	   : "g1" __AND_CLOBBER_CC) +#define UDIV_TIME 37 +#define count_leading_zeros(count, x) \ +  do {                                                                  \ +  __asm__ ("scan %1,1,%0"                                               \ +	   : "=r" ((USItype) (count))                                   \ +	   : "r" ((USItype) (x)));					\ +  } while (0) +/* Early sparclites return 63 for an argument of 0, but they warn that future +   implementations might change this.  Therefore, leave COUNT_LEADING_ZEROS_0 +   undefined.  */ +#else +/* SPARC without integer multiplication and divide instructions. +   (i.e. at least Sun4/20,40,60,65,75,110,260,280,330,360,380,470,490) */ +#define umul_ppmm(w1, w0, u, v) \ +  __asm__ ("! Inlined umul_ppmm\n"					\ +"	wr	%%g0,%2,%%y	! SPARC has 0-3 delay insn after a wr\n"\ +"	sra	%3,31,%%o5	! Don't move this insn\n"		\ +"	and	%2,%%o5,%%o5	! Don't move this insn\n"		\ +"	andcc	%%g0,0,%%g1	! Don't move this insn\n"		\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,%3,%%g1\n"						\ +"	mulscc	%%g1,0,%%g1\n"						\ +"	add	%%g1,%%o5,%0\n"						\ +"	rd	%%y,%1"							\ +	   : "=r" ((USItype) (w1)),					\ +	     "=r" ((USItype) (w0))					\ +	   : "%rI" ((USItype) (u)),					\ +	     "r" ((USItype) (v))						\ +	   : "g1", "o5" __AND_CLOBBER_CC) +#define UMUL_TIME 39		/* 39 instructions */ +/* It's quite necessary to add this much assembler for the sparc. +   The default udiv_qrnnd (in C) is more than 10 times slower!  */ +#define udiv_qrnnd(__q, __r, __n1, __n0, __d) \ +  __asm__ ("! Inlined udiv_qrnnd\n"					\ +"	mov	32,%%g1\n"						\ +"	subcc	%1,%2,%%g0\n"						\ +"1:	bcs	5f\n"							\ +"	 addxcc %0,%0,%0	! shift n1n0 and a q-bit in lsb\n"	\ +"	sub	%1,%2,%1	! this kills msb of n\n"		\ +"	addx	%1,%1,%1	! so this can't give carry\n"		\ +"	subcc	%%g1,1,%%g1\n"						\ +"2:	bne	1b\n"							\ +"	 subcc	%1,%2,%%g0\n"						\ +"	bcs	3f\n"							\ +"	 addxcc %0,%0,%0	! shift n1n0 and a q-bit in lsb\n"	\ +"	b	3f\n"							\ +"	 sub	%1,%2,%1	! this kills msb of n\n"		\ +"4:	sub	%1,%2,%1\n"						\ +"5:	addxcc	%1,%1,%1\n"						\ +"	bcc	2b\n"							\ +"	 subcc	%%g1,1,%%g1\n"						\ +"! Got carry from n.  Subtract next step to cancel this carry.\n"	\ +"	bne	4b\n"							\ +"	 addcc	%0,%0,%0	! shift n1n0 and a 0-bit in lsb\n"	\ +"	sub	%1,%2,%1\n"						\ +"3:	xnor	%0,0,%0\n"						\ +"	! End of inline udiv_qrnnd"					\ +	   : "=&r" ((USItype) (__q)),					\ +	     "=&r" ((USItype) (__r))					\ +	   : "r" ((USItype) (__d)),					\ +	     "1" ((USItype) (__n1)),					\ +	     "0" ((USItype) (__n0)) : "g1" __AND_CLOBBER_CC) +#define UDIV_TIME (3+7*32)	/* 7 instructions/iteration. 32 iterations.  */ +#endif /* __sparclite__ */ +#endif /* __sparc_v8__ */ +#endif /* __sparc_v9__ */ +#endif /* sparc32 */ + +#if ((defined (__sparc__) && defined (__arch64__)) || defined (__sparcv9)) \ +    && W_TYPE_SIZE == 64 +#define add_ssaaaa(sh, sl, ah, al, bh, bl)				\ +  do {									\ +    UDItype __carry = 0;						\ +    __asm__ ("addcc\t%r5,%6,%1\n\t"					\ +	     "add\t%r3,%4,%0\n\t"					\ +	     "movcs\t%%xcc, 1, %2\n\t"					\ +	     "add\t%0, %2, %0"						\ +	     : "=r" ((UDItype)(sh)),				      	\ +	       "=&r" ((UDItype)(sl)),				      	\ +	       "+r" (__carry)				      		\ +	     : "%rJ" ((UDItype)(ah)),				     	\ +	       "rI" ((UDItype)(bh)),				      	\ +	       "%rJ" ((UDItype)(al)),				     	\ +	       "rI" ((UDItype)(bl))				       	\ +	     __CLOBBER_CC);						\ +  } while (0) + +#define sub_ddmmss(sh, sl, ah, al, bh, bl)				\ +  do {									\ +    UDItype __carry = 0;						\ +    __asm__ ("subcc\t%r5,%6,%1\n\t"					\ +	     "sub\t%r3,%4,%0\n\t"					\ +	     "movcs\t%%xcc, 1, %2\n\t"					\ +	     "sub\t%0, %2, %0"						\ +	     : "=r" ((UDItype)(sh)),				      	\ +	       "=&r" ((UDItype)(sl)),				      	\ +	       "+r" (__carry)				      		\ +	     : "%rJ" ((UDItype)(ah)),				     	\ +	       "rI" ((UDItype)(bh)),				      	\ +	       "%rJ" ((UDItype)(al)),				     	\ +	       "rI" ((UDItype)(bl))				       	\ +	     __CLOBBER_CC);						\ +  } while (0) + +#define umul_ppmm(wh, wl, u, v)						\ +  do {									\ +	  UDItype tmp1, tmp2, tmp3, tmp4;				\ +	  __asm__ __volatile__ (					\ +		   "srl %7,0,%3\n\t"					\ +		   "mulx %3,%6,%1\n\t"					\ +		   "srlx %6,32,%2\n\t"					\ +		   "mulx %2,%3,%4\n\t"					\ +		   "sllx %4,32,%5\n\t"					\ +		   "srl %6,0,%3\n\t"					\ +		   "sub %1,%5,%5\n\t"					\ +		   "srlx %5,32,%5\n\t"					\ +		   "addcc %4,%5,%4\n\t"					\ +		   "srlx %7,32,%5\n\t"					\ +		   "mulx %3,%5,%3\n\t"					\ +		   "mulx %2,%5,%5\n\t"					\ +		   "sethi %%hi(0x80000000),%2\n\t"			\ +		   "addcc %4,%3,%4\n\t"					\ +		   "srlx %4,32,%4\n\t"					\ +		   "add %2,%2,%2\n\t"					\ +		   "movcc %%xcc,%%g0,%2\n\t"				\ +		   "addcc %5,%4,%5\n\t"					\ +		   "sllx %3,32,%3\n\t"					\ +		   "add %1,%3,%1\n\t"					\ +		   "add %5,%2,%0"					\ +	   : "=r" ((UDItype)(wh)),					\ +	     "=&r" ((UDItype)(wl)),					\ +	     "=&r" (tmp1), "=&r" (tmp2), "=&r" (tmp3), "=&r" (tmp4)	\ +	   : "r" ((UDItype)(u)),					\ +	     "r" ((UDItype)(v))						\ +	   __CLOBBER_CC);						\ +  } while (0) +#define UMUL_TIME 96 +#define UDIV_TIME 230 +#endif /* sparc64 */ + +#if defined (__vax__) && W_TYPE_SIZE == 32 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  __asm__ ("addl2 %5,%1\n\tadwc %3,%0"					\ +	   : "=g" ((USItype) (sh)),					\ +	     "=&g" ((USItype) (sl))					\ +	   : "%0" ((USItype) (ah)),					\ +	     "g" ((USItype) (bh)),					\ +	     "%1" ((USItype) (al)),					\ +	     "g" ((USItype) (bl))) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  __asm__ ("subl2 %5,%1\n\tsbwc %3,%0"					\ +	   : "=g" ((USItype) (sh)),					\ +	     "=&g" ((USItype) (sl))					\ +	   : "0" ((USItype) (ah)),					\ +	     "g" ((USItype) (bh)),					\ +	     "1" ((USItype) (al)),					\ +	     "g" ((USItype) (bl))) +#define umul_ppmm(xh, xl, m0, m1) \ +  do {									\ +    union {								\ +	UDItype __ll;							\ +	struct {USItype __l, __h;} __i;					\ +      } __xx;								\ +    USItype __m0 = (m0), __m1 = (m1);					\ +    __asm__ ("emul %1,%2,$0,%0"						\ +	     : "=r" (__xx.__ll)						\ +	     : "g" (__m0),						\ +	       "g" (__m1));						\ +    (xh) = __xx.__i.__h;						\ +    (xl) = __xx.__i.__l;						\ +    (xh) += ((((SItype) __m0 >> 31) & __m1)				\ +	     + (((SItype) __m1 >> 31) & __m0));				\ +  } while (0) +#define sdiv_qrnnd(q, r, n1, n0, d) \ +  do {									\ +    union {DItype __ll;							\ +	   struct {SItype __l, __h;} __i;				\ +	  } __xx;							\ +    __xx.__i.__h = n1; __xx.__i.__l = n0;				\ +    __asm__ ("ediv %3,%2,%0,%1"						\ +	     : "=g" (q), "=g" (r)					\ +	     : "g" (__xx.__ll), "g" (d));				\ +  } while (0) +#endif /* __vax__ */ + +#ifdef _TMS320C6X +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  do									\ +    {									\ +      UDItype __ll;							\ +      __asm__ ("addu .l1 %1, %2, %0"					\ +	       : "=a" (__ll) : "a" (al), "a" (bl));			\ +      (sl) = (USItype)__ll;						\ +      (sh) = ((USItype)(__ll >> 32)) + (ah) + (bh);			\ +    }									\ +  while (0) + +#ifdef _TMS320C6400_PLUS +#define __umulsidi3(u,v) ((UDItype)(USItype)u*(USItype)v) +#define umul_ppmm(w1, w0, u, v)						\ +  do {									\ +    UDItype __x = (UDItype) (USItype) (u) * (USItype) (v);		\ +    (w1) = (USItype) (__x >> 32);					\ +    (w0) = (USItype) (__x);						\ +  } while (0) +#endif  /* _TMS320C6400_PLUS */ + +#define count_leading_zeros(count, x)	((count) = __builtin_clz (x)) +#ifdef _TMS320C6400 +#define count_trailing_zeros(count, x)	((count) = __builtin_ctz (x)) +#endif +#define UMUL_TIME 4 +#define UDIV_TIME 40 +#endif /* _TMS320C6X */ + +#if defined (__xtensa__) && W_TYPE_SIZE == 32 +/* This code is not Xtensa-configuration-specific, so rely on the compiler +   to expand builtin functions depending on what configuration features +   are available.  This avoids library calls when the operation can be +   performed in-line.  */ +#define umul_ppmm(w1, w0, u, v)						\ +  do {									\ +    DWunion __w;							\ +    __w.ll = __builtin_umulsidi3 (u, v);				\ +    w1 = __w.s.high;							\ +    w0 = __w.s.low;							\ +  } while (0) +#define __umulsidi3(u, v)		__builtin_umulsidi3 (u, v) +#define count_leading_zeros(COUNT, X)	((COUNT) = __builtin_clz (X)) +#define count_trailing_zeros(COUNT, X)	((COUNT) = __builtin_ctz (X)) +#endif /* __xtensa__ */ + +#if defined xstormy16 +extern UHItype __stormy16_count_leading_zeros (UHItype); +#define count_leading_zeros(count, x)					\ +  do									\ +    {									\ +      UHItype size;							\ +									\ +      /* We assume that W_TYPE_SIZE is a multiple of 16...  */		\ +      for ((count) = 0, size = W_TYPE_SIZE; size; size -= 16)		\ +	{								\ +	  UHItype c;							\ +									\ +	  c = __clzhi2 ((x) >> (size - 16));				\ +	  (count) += c;							\ +	  if (c != 16)							\ +	    break;							\ +	}								\ +    }									\ +  while (0) +#define COUNT_LEADING_ZEROS_0 W_TYPE_SIZE +#endif + +#if defined (__z8000__) && W_TYPE_SIZE == 16 +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  __asm__ ("add	%H1,%H5\n\tadc	%H0,%H3"				\ +	   : "=r" ((unsigned int)(sh)),					\ +	     "=&r" ((unsigned int)(sl))					\ +	   : "%0" ((unsigned int)(ah)),					\ +	     "r" ((unsigned int)(bh)),					\ +	     "%1" ((unsigned int)(al)),					\ +	     "rQR" ((unsigned int)(bl))) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  __asm__ ("sub	%H1,%H5\n\tsbc	%H0,%H3"				\ +	   : "=r" ((unsigned int)(sh)),					\ +	     "=&r" ((unsigned int)(sl))					\ +	   : "0" ((unsigned int)(ah)),					\ +	     "r" ((unsigned int)(bh)),					\ +	     "1" ((unsigned int)(al)),					\ +	     "rQR" ((unsigned int)(bl))) +#define umul_ppmm(xh, xl, m0, m1) \ +  do {									\ +    union {long int __ll;						\ +	   struct {unsigned int __h, __l;} __i;				\ +	  } __xx;							\ +    unsigned int __m0 = (m0), __m1 = (m1);				\ +    __asm__ ("mult	%S0,%H3"					\ +	     : "=r" (__xx.__i.__h),					\ +	       "=r" (__xx.__i.__l)					\ +	     : "%1" (__m0),						\ +	       "rQR" (__m1));						\ +    (xh) = __xx.__i.__h; (xl) = __xx.__i.__l;				\ +    (xh) += ((((signed int) __m0 >> 15) & __m1)				\ +	     + (((signed int) __m1 >> 15) & __m0));			\ +  } while (0) +#endif /* __z8000__ */ + +#endif /* __GNUC__ */ + +/* If this machine has no inline assembler, use C macros.  */ + +#if !defined (add_ssaaaa) +#define add_ssaaaa(sh, sl, ah, al, bh, bl) \ +  do {									\ +    UWtype __x;								\ +    __x = (al) + (bl);							\ +    (sh) = (ah) + (bh) + (__x < (al));					\ +    (sl) = __x;								\ +  } while (0) +#endif + +#if !defined (sub_ddmmss) +#define sub_ddmmss(sh, sl, ah, al, bh, bl) \ +  do {									\ +    UWtype __x;								\ +    __x = (al) - (bl);							\ +    (sh) = (ah) - (bh) - (__x > (al));					\ +    (sl) = __x;								\ +  } while (0) +#endif + +/* If we lack umul_ppmm but have smul_ppmm, define umul_ppmm in terms of +   smul_ppmm.  */ +#if !defined (umul_ppmm) && defined (smul_ppmm) +#define umul_ppmm(w1, w0, u, v)						\ +  do {									\ +    UWtype __w1;							\ +    UWtype __xm0 = (u), __xm1 = (v);					\ +    smul_ppmm (__w1, w0, __xm0, __xm1);					\ +    (w1) = __w1 + (-(__xm0 >> (W_TYPE_SIZE - 1)) & __xm1)		\ +		+ (-(__xm1 >> (W_TYPE_SIZE - 1)) & __xm0);		\ +  } while (0) +#endif + +/* If we still don't have umul_ppmm, define it using plain C.  */ +#if !defined (umul_ppmm) +#define umul_ppmm(w1, w0, u, v)						\ +  do {									\ +    UWtype __x0, __x1, __x2, __x3;					\ +    UHWtype __ul, __vl, __uh, __vh;					\ +									\ +    __ul = __ll_lowpart (u);						\ +    __uh = __ll_highpart (u);						\ +    __vl = __ll_lowpart (v);						\ +    __vh = __ll_highpart (v);						\ +									\ +    __x0 = (UWtype) __ul * __vl;					\ +    __x1 = (UWtype) __ul * __vh;					\ +    __x2 = (UWtype) __uh * __vl;					\ +    __x3 = (UWtype) __uh * __vh;					\ +									\ +    __x1 += __ll_highpart (__x0);/* this can't give carry */		\ +    __x1 += __x2;		/* but this indeed can */		\ +    if (__x1 < __x2)		/* did we get it? */			\ +      __x3 += __ll_B;		/* yes, add it in the proper pos.  */	\ +									\ +    (w1) = __x3 + __ll_highpart (__x1);					\ +    (w0) = __ll_lowpart (__x1) * __ll_B + __ll_lowpart (__x0);		\ +  } while (0) +#endif + +#if !defined (__umulsidi3) +#define __umulsidi3(u, v) \ +  ({DWunion __w;							\ +    umul_ppmm (__w.s.high, __w.s.low, u, v);				\ +    __w.ll; }) +#endif + +/* Define this unconditionally, so it can be used for debugging.  */ +#define __udiv_qrnnd_c(q, r, n1, n0, d) \ +  do {									\ +    UWtype __d1, __d0, __q1, __q0;					\ +    UWtype __r1, __r0, __m;						\ +    __d1 = __ll_highpart (d);						\ +    __d0 = __ll_lowpart (d);						\ +									\ +    __r1 = (n1) % __d1;							\ +    __q1 = (n1) / __d1;							\ +    __m = (UWtype) __q1 * __d0;						\ +    __r1 = __r1 * __ll_B | __ll_highpart (n0);				\ +    if (__r1 < __m)							\ +      {									\ +	__q1--, __r1 += (d);						\ +	if (__r1 >= (d)) /* i.e. we didn't get carry when adding to __r1 */\ +	  if (__r1 < __m)						\ +	    __q1--, __r1 += (d);					\ +      }									\ +    __r1 -= __m;							\ +									\ +    __r0 = __r1 % __d1;							\ +    __q0 = __r1 / __d1;							\ +    __m = (UWtype) __q0 * __d0;						\ +    __r0 = __r0 * __ll_B | __ll_lowpart (n0);				\ +    if (__r0 < __m)							\ +      {									\ +	__q0--, __r0 += (d);						\ +	if (__r0 >= (d))						\ +	  if (__r0 < __m)						\ +	    __q0--, __r0 += (d);					\ +      }									\ +    __r0 -= __m;							\ +									\ +    (q) = (UWtype) __q1 * __ll_B | __q0;				\ +    (r) = __r0;								\ +  } while (0) + +/* If the processor has no udiv_qrnnd but sdiv_qrnnd, go through +   __udiv_w_sdiv (defined in libgcc or elsewhere).  */ +#if !defined (udiv_qrnnd) && defined (sdiv_qrnnd) +#define udiv_qrnnd(q, r, nh, nl, d) \ +  do {									\ +    extern UWtype __udiv_w_sdiv (UWtype *, UWtype, UWtype, UWtype);	\ +    UWtype __r;								\ +    (q) = __udiv_w_sdiv (&__r, nh, nl, d);				\ +    (r) = __r;								\ +  } while (0) +#endif + +/* If udiv_qrnnd was not defined for this processor, use __udiv_qrnnd_c.  */ +#if !defined (udiv_qrnnd) +#define UDIV_NEEDS_NORMALIZATION 1 +#define udiv_qrnnd __udiv_qrnnd_c +#endif + +#if !defined (count_leading_zeros) +#define count_leading_zeros(count, x) \ +  do {									\ +    UWtype __xr = (x);							\ +    UWtype __a;								\ +									\ +    if (W_TYPE_SIZE <= 32)						\ +      {									\ +	__a = __xr < ((UWtype)1<<2*__BITS4)				\ +	  ? (__xr < ((UWtype)1<<__BITS4) ? 0 : __BITS4)			\ +	  : (__xr < ((UWtype)1<<3*__BITS4) ?  2*__BITS4 : 3*__BITS4);	\ +      }									\ +    else								\ +      {									\ +	for (__a = W_TYPE_SIZE - 8; __a > 0; __a -= 8)			\ +	  if (((__xr >> __a) & 0xff) != 0)				\ +	    break;							\ +      }									\ +									\ +    (count) = W_TYPE_SIZE - (__clz_tab[__xr >> __a] + __a);		\ +  } while (0) +#define COUNT_LEADING_ZEROS_0 W_TYPE_SIZE +#endif + +#if !defined (count_trailing_zeros) +/* Define count_trailing_zeros using count_leading_zeros.  The latter might be +   defined in asm, but if it is not, the C version above is good enough.  */ +#define count_trailing_zeros(count, x) \ +  do {									\ +    UWtype __ctz_x = (x);						\ +    UWtype __ctz_c;							\ +    count_leading_zeros (__ctz_c, __ctz_x & -__ctz_x);			\ +    (count) = W_TYPE_SIZE - 1 - __ctz_c;				\ +  } while (0) +#endif + +#ifndef UDIV_NEEDS_NORMALIZATION +#define UDIV_NEEDS_NORMALIZATION 0 +#endif diff --git a/libc/sysdeps/linux/sparc64/soft-fp/op-1.h b/libc/sysdeps/linux/sparc64/soft-fp/op-1.h new file mode 100644 index 000000000..e1e39664d --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/op-1.h @@ -0,0 +1,369 @@ +/* Software floating-point emulation. +   Basic one-word fraction declaration and manipulation. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com), +		  Jakub Jelinek (jj@ultra.linux.cz), +		  David S. Miller (davem@redhat.com) and +		  Peter Maydell (pmaydell@chiark.greenend.org.uk). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 of the License, or (at your option) any later version. + +   In addition to the permissions in the GNU Lesser General Public +   License, the Free Software Foundation gives you unlimited +   permission to link the compiled version of this file into +   combinations with other programs, and to distribute those +   combinations without any restriction coming from the use of this +   file.  (The Lesser General Public License restrictions do apply in +   other respects; for example, they cover modification of the file, +   and distribution when not linked into a combine executable.) + +   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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef SOFT_FP_OP_1_H +#define SOFT_FP_OP_1_H	1 + +#define _FP_FRAC_DECL_1(X)	_FP_W_TYPE X##_f _FP_ZERO_INIT +#define _FP_FRAC_COPY_1(D, S)	(D##_f = S##_f) +#define _FP_FRAC_SET_1(X, I)	(X##_f = I) +#define _FP_FRAC_HIGH_1(X)	(X##_f) +#define _FP_FRAC_LOW_1(X)	(X##_f) +#define _FP_FRAC_WORD_1(X, w)	(X##_f) + +#define _FP_FRAC_ADDI_1(X, I)	(X##_f += I) +#define _FP_FRAC_SLL_1(X, N)			\ +  do						\ +    {						\ +      if (__builtin_constant_p (N) && (N) == 1)	\ +	X##_f += X##_f;				\ +      else					\ +	X##_f <<= (N);				\ +    }						\ +  while (0) +#define _FP_FRAC_SRL_1(X, N)	(X##_f >>= N) + +/* Right shift with sticky-lsb.  */ +#define _FP_FRAC_SRST_1(X, S, N, sz)	__FP_FRAC_SRST_1 (X##_f, S, (N), (sz)) +#define _FP_FRAC_SRS_1(X, N, sz)	__FP_FRAC_SRS_1 (X##_f, (N), (sz)) + +#define __FP_FRAC_SRST_1(X, S, N, sz)			\ +  do							\ +    {							\ +      S = (__builtin_constant_p (N) && (N) == 1		\ +	   ? X & 1					\ +	   : (X << (_FP_W_TYPE_SIZE - (N))) != 0);	\ +      X = X >> (N);					\ +    }							\ +  while (0) + +#define __FP_FRAC_SRS_1(X, N, sz)				\ +  (X = (X >> (N) | (__builtin_constant_p (N) && (N) == 1	\ +		    ? X & 1					\ +		    : (X << (_FP_W_TYPE_SIZE - (N))) != 0))) + +#define _FP_FRAC_ADD_1(R, X, Y)	(R##_f = X##_f + Y##_f) +#define _FP_FRAC_SUB_1(R, X, Y)	(R##_f = X##_f - Y##_f) +#define _FP_FRAC_DEC_1(X, Y)	(X##_f -= Y##_f) +#define _FP_FRAC_CLZ_1(z, X)	__FP_CLZ ((z), X##_f) + +/* Predicates.  */ +#define _FP_FRAC_NEGP_1(X)	((_FP_WS_TYPE) X##_f < 0) +#define _FP_FRAC_ZEROP_1(X)	(X##_f == 0) +#define _FP_FRAC_OVERP_1(fs, X)	(X##_f & _FP_OVERFLOW_##fs) +#define _FP_FRAC_CLEAR_OVERP_1(fs, X)	(X##_f &= ~_FP_OVERFLOW_##fs) +#define _FP_FRAC_HIGHBIT_DW_1(fs, X)	(X##_f & _FP_HIGHBIT_DW_##fs) +#define _FP_FRAC_EQ_1(X, Y)	(X##_f == Y##_f) +#define _FP_FRAC_GE_1(X, Y)	(X##_f >= Y##_f) +#define _FP_FRAC_GT_1(X, Y)	(X##_f > Y##_f) + +#define _FP_ZEROFRAC_1		0 +#define _FP_MINFRAC_1		1 +#define _FP_MAXFRAC_1		(~(_FP_WS_TYPE) 0) + +/* Unpack the raw bits of a native fp value.  Do not classify or +   normalize the data.  */ + +#define _FP_UNPACK_RAW_1(fs, X, val)			\ +  do							\ +    {							\ +      union _FP_UNION_##fs _FP_UNPACK_RAW_1_flo;	\ +      _FP_UNPACK_RAW_1_flo.flt = (val);			\ +							\ +      X##_f = _FP_UNPACK_RAW_1_flo.bits.frac;		\ +      X##_e = _FP_UNPACK_RAW_1_flo.bits.exp;		\ +      X##_s = _FP_UNPACK_RAW_1_flo.bits.sign;		\ +    }							\ +  while (0) + +#define _FP_UNPACK_RAW_1_P(fs, X, val)			\ +  do							\ +    {							\ +      union _FP_UNION_##fs *_FP_UNPACK_RAW_1_P_flo	\ +	= (union _FP_UNION_##fs *) (val);		\ +							\ +      X##_f = _FP_UNPACK_RAW_1_P_flo->bits.frac;	\ +      X##_e = _FP_UNPACK_RAW_1_P_flo->bits.exp;		\ +      X##_s = _FP_UNPACK_RAW_1_P_flo->bits.sign;	\ +    }							\ +  while (0) + +/* Repack the raw bits of a native fp value.  */ + +#define _FP_PACK_RAW_1(fs, val, X)		\ +  do						\ +    {						\ +      union _FP_UNION_##fs _FP_PACK_RAW_1_flo;	\ +						\ +      _FP_PACK_RAW_1_flo.bits.frac = X##_f;	\ +      _FP_PACK_RAW_1_flo.bits.exp  = X##_e;	\ +      _FP_PACK_RAW_1_flo.bits.sign = X##_s;	\ +						\ +      (val) = _FP_PACK_RAW_1_flo.flt;		\ +    }						\ +  while (0) + +#define _FP_PACK_RAW_1_P(fs, val, X)			\ +  do							\ +    {							\ +      union _FP_UNION_##fs *_FP_PACK_RAW_1_P_flo	\ +	= (union _FP_UNION_##fs *) (val);		\ +							\ +      _FP_PACK_RAW_1_P_flo->bits.frac = X##_f;		\ +      _FP_PACK_RAW_1_P_flo->bits.exp  = X##_e;		\ +      _FP_PACK_RAW_1_P_flo->bits.sign = X##_s;		\ +    }							\ +  while (0) + + +/* Multiplication algorithms: */ + +/* Basic.  Assuming the host word size is >= 2*FRACBITS, we can do the +   multiplication immediately.  */ + +#define _FP_MUL_MEAT_DW_1_imm(wfracbits, R, X, Y)	\ +  do							\ +    {							\ +      R##_f = X##_f * Y##_f;				\ +    }							\ +  while (0) + +#define _FP_MUL_MEAT_1_imm(wfracbits, R, X, Y)				\ +  do									\ +    {									\ +      _FP_MUL_MEAT_DW_1_imm ((wfracbits), R, X, Y);			\ +      /* Normalize since we know where the msb of the multiplicands	\ +	 were (bit B), we know that the msb of the of the product is	\ +	 at either 2B or 2B-1.  */					\ +      _FP_FRAC_SRS_1 (R, (wfracbits)-1, 2*(wfracbits));			\ +    }									\ +  while (0) + +/* Given a 1W * 1W => 2W primitive, do the extended multiplication.  */ + +#define _FP_MUL_MEAT_DW_1_wide(wfracbits, R, X, Y, doit)	\ +  do								\ +    {								\ +      doit (R##_f1, R##_f0, X##_f, Y##_f);			\ +    }								\ +  while (0) + +#define _FP_MUL_MEAT_1_wide(wfracbits, R, X, Y, doit)			\ +  do									\ +    {									\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_1_wide_Z);				\ +      _FP_MUL_MEAT_DW_1_wide ((wfracbits), _FP_MUL_MEAT_1_wide_Z,	\ +			      X, Y, doit);				\ +      /* Normalize since we know where the msb of the multiplicands	\ +	 were (bit B), we know that the msb of the of the product is	\ +	 at either 2B or 2B-1.  */					\ +      _FP_FRAC_SRS_2 (_FP_MUL_MEAT_1_wide_Z, (wfracbits)-1,		\ +		      2*(wfracbits));					\ +      R##_f = _FP_MUL_MEAT_1_wide_Z_f0;					\ +    }									\ +  while (0) + +/* Finally, a simple widening multiply algorithm.  What fun!  */ + +#define _FP_MUL_MEAT_DW_1_hard(wfracbits, R, X, Y)			\ +  do									\ +    {									\ +      _FP_W_TYPE _FP_MUL_MEAT_DW_1_hard_xh, _FP_MUL_MEAT_DW_1_hard_xl;	\ +      _FP_W_TYPE _FP_MUL_MEAT_DW_1_hard_yh, _FP_MUL_MEAT_DW_1_hard_yl;	\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_DW_1_hard_a);			\ +									\ +      /* Split the words in half.  */					\ +      _FP_MUL_MEAT_DW_1_hard_xh = X##_f >> (_FP_W_TYPE_SIZE/2);		\ +      _FP_MUL_MEAT_DW_1_hard_xl						\ +	= X##_f & (((_FP_W_TYPE) 1 << (_FP_W_TYPE_SIZE/2)) - 1);	\ +      _FP_MUL_MEAT_DW_1_hard_yh = Y##_f >> (_FP_W_TYPE_SIZE/2);		\ +      _FP_MUL_MEAT_DW_1_hard_yl						\ +	= Y##_f & (((_FP_W_TYPE) 1 << (_FP_W_TYPE_SIZE/2)) - 1);	\ +									\ +      /* Multiply the pieces.  */					\ +      R##_f0 = _FP_MUL_MEAT_DW_1_hard_xl * _FP_MUL_MEAT_DW_1_hard_yl;	\ +      _FP_MUL_MEAT_DW_1_hard_a_f0					\ +	= _FP_MUL_MEAT_DW_1_hard_xh * _FP_MUL_MEAT_DW_1_hard_yl;	\ +      _FP_MUL_MEAT_DW_1_hard_a_f1					\ +	= _FP_MUL_MEAT_DW_1_hard_xl * _FP_MUL_MEAT_DW_1_hard_yh;	\ +      R##_f1 = _FP_MUL_MEAT_DW_1_hard_xh * _FP_MUL_MEAT_DW_1_hard_yh;	\ +									\ +      /* Reassemble into two full words.  */				\ +      if ((_FP_MUL_MEAT_DW_1_hard_a_f0 += _FP_MUL_MEAT_DW_1_hard_a_f1)	\ +	  < _FP_MUL_MEAT_DW_1_hard_a_f1)				\ +	R##_f1 += (_FP_W_TYPE) 1 << (_FP_W_TYPE_SIZE/2);		\ +      _FP_MUL_MEAT_DW_1_hard_a_f1					\ +	= _FP_MUL_MEAT_DW_1_hard_a_f0 >> (_FP_W_TYPE_SIZE/2);		\ +      _FP_MUL_MEAT_DW_1_hard_a_f0					\ +	= _FP_MUL_MEAT_DW_1_hard_a_f0 << (_FP_W_TYPE_SIZE/2);		\ +      _FP_FRAC_ADD_2 (R, R, _FP_MUL_MEAT_DW_1_hard_a);			\ +    }									\ +  while (0) + +#define _FP_MUL_MEAT_1_hard(wfracbits, R, X, Y)			\ +  do								\ +    {								\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_1_hard_z);			\ +      _FP_MUL_MEAT_DW_1_hard ((wfracbits),			\ +			      _FP_MUL_MEAT_1_hard_z, X, Y);	\ +								\ +      /* Normalize.  */						\ +      _FP_FRAC_SRS_2 (_FP_MUL_MEAT_1_hard_z,			\ +		      (wfracbits) - 1, 2*(wfracbits));		\ +      R##_f = _FP_MUL_MEAT_1_hard_z_f0;				\ +    }								\ +  while (0) + + +/* Division algorithms: */ + +/* Basic.  Assuming the host word size is >= 2*FRACBITS, we can do the +   division immediately.  Give this macro either _FP_DIV_HELP_imm for +   C primitives or _FP_DIV_HELP_ldiv for the ISO function.  Which you +   choose will depend on what the compiler does with divrem4.  */ + +#define _FP_DIV_MEAT_1_imm(fs, R, X, Y, doit)				\ +  do									\ +    {									\ +      _FP_W_TYPE _FP_DIV_MEAT_1_imm_q, _FP_DIV_MEAT_1_imm_r;		\ +      X##_f <<= (X##_f < Y##_f						\ +		 ? R##_e--, _FP_WFRACBITS_##fs				\ +		 : _FP_WFRACBITS_##fs - 1);				\ +      doit (_FP_DIV_MEAT_1_imm_q, _FP_DIV_MEAT_1_imm_r, X##_f, Y##_f);	\ +      R##_f = _FP_DIV_MEAT_1_imm_q | (_FP_DIV_MEAT_1_imm_r != 0);	\ +    }									\ +  while (0) + +/* GCC's longlong.h defines a 2W / 1W => (1W,1W) primitive udiv_qrnnd +   that may be useful in this situation.  This first is for a primitive +   that requires normalization, the second for one that does not.  Look +   for UDIV_NEEDS_NORMALIZATION to tell which your machine needs.  */ + +#define _FP_DIV_MEAT_1_udiv_norm(fs, R, X, Y)				\ +  do									\ +    {									\ +      _FP_W_TYPE _FP_DIV_MEAT_1_udiv_norm_nh;				\ +      _FP_W_TYPE _FP_DIV_MEAT_1_udiv_norm_nl;				\ +      _FP_W_TYPE _FP_DIV_MEAT_1_udiv_norm_q;				\ +      _FP_W_TYPE _FP_DIV_MEAT_1_udiv_norm_r;				\ +      _FP_W_TYPE _FP_DIV_MEAT_1_udiv_norm_y;				\ +									\ +      /* Normalize Y -- i.e. make the most significant bit set.  */	\ +      _FP_DIV_MEAT_1_udiv_norm_y = Y##_f << _FP_WFRACXBITS_##fs;	\ +									\ +      /* Shift X op correspondingly high, that is, up one full word.  */ \ +      if (X##_f < Y##_f)						\ +	{								\ +	  R##_e--;							\ +	  _FP_DIV_MEAT_1_udiv_norm_nl = 0;				\ +	  _FP_DIV_MEAT_1_udiv_norm_nh = X##_f;				\ +	}								\ +      else								\ +	{								\ +	  _FP_DIV_MEAT_1_udiv_norm_nl = X##_f << (_FP_W_TYPE_SIZE - 1);	\ +	  _FP_DIV_MEAT_1_udiv_norm_nh = X##_f >> 1;			\ +	}								\ +									\ +      udiv_qrnnd (_FP_DIV_MEAT_1_udiv_norm_q,				\ +		  _FP_DIV_MEAT_1_udiv_norm_r,				\ +		  _FP_DIV_MEAT_1_udiv_norm_nh,				\ +		  _FP_DIV_MEAT_1_udiv_norm_nl,				\ +		  _FP_DIV_MEAT_1_udiv_norm_y);				\ +      R##_f = (_FP_DIV_MEAT_1_udiv_norm_q				\ +	       | (_FP_DIV_MEAT_1_udiv_norm_r != 0));			\ +    }									\ +  while (0) + +#define _FP_DIV_MEAT_1_udiv(fs, R, X, Y)				\ +  do									\ +    {									\ +      _FP_W_TYPE _FP_DIV_MEAT_1_udiv_nh, _FP_DIV_MEAT_1_udiv_nl;	\ +      _FP_W_TYPE _FP_DIV_MEAT_1_udiv_q, _FP_DIV_MEAT_1_udiv_r;		\ +      if (X##_f < Y##_f)						\ +	{								\ +	  R##_e--;							\ +	  _FP_DIV_MEAT_1_udiv_nl = X##_f << _FP_WFRACBITS_##fs;		\ +	  _FP_DIV_MEAT_1_udiv_nh = X##_f >> _FP_WFRACXBITS_##fs;	\ +	}								\ +      else								\ +	{								\ +	  _FP_DIV_MEAT_1_udiv_nl = X##_f << (_FP_WFRACBITS_##fs - 1);	\ +	  _FP_DIV_MEAT_1_udiv_nh = X##_f >> (_FP_WFRACXBITS_##fs + 1);	\ +	}								\ +      udiv_qrnnd (_FP_DIV_MEAT_1_udiv_q, _FP_DIV_MEAT_1_udiv_r,		\ +		  _FP_DIV_MEAT_1_udiv_nh, _FP_DIV_MEAT_1_udiv_nl,	\ +		  Y##_f);						\ +      R##_f = _FP_DIV_MEAT_1_udiv_q | (_FP_DIV_MEAT_1_udiv_r != 0);	\ +    }									\ +  while (0) + + +/* Square root algorithms: +   We have just one right now, maybe Newton approximation +   should be added for those machines where division is fast.  */ + +#define _FP_SQRT_MEAT_1(R, S, T, X, q)		\ +  do						\ +    {						\ +      while ((q) != _FP_WORK_ROUND)		\ +	{					\ +	  T##_f = S##_f + (q);			\ +	  if (T##_f <= X##_f)			\ +	    {					\ +	      S##_f = T##_f + (q);		\ +	      X##_f -= T##_f;			\ +	      R##_f += (q);			\ +	    }					\ +	  _FP_FRAC_SLL_1 (X, 1);		\ +	  (q) >>= 1;				\ +	}					\ +      if (X##_f)				\ +	{					\ +	  if (S##_f < X##_f)			\ +	    R##_f |= _FP_WORK_ROUND;		\ +	  R##_f |= _FP_WORK_STICKY;		\ +	}					\ +    }						\ +  while (0) + +/* Assembly/disassembly for converting to/from integral types. +   No shifting or overflow handled here.  */ + +#define _FP_FRAC_ASSEMBLE_1(r, X, rsize)	((r) = X##_f) +#define _FP_FRAC_DISASSEMBLE_1(X, r, rsize)	(X##_f = (r)) + + +/* Convert FP values between word sizes.  */ + +#define _FP_FRAC_COPY_1_1(D, S)		(D##_f = S##_f) + +#endif /* !SOFT_FP_OP_1_H */ diff --git a/libc/sysdeps/linux/sparc64/soft-fp/op-2.h b/libc/sysdeps/linux/sparc64/soft-fp/op-2.h new file mode 100644 index 000000000..c010afa3e --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/op-2.h @@ -0,0 +1,705 @@ +/* Software floating-point emulation. +   Basic two-word fraction declaration and manipulation. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com), +		  Jakub Jelinek (jj@ultra.linux.cz), +		  David S. Miller (davem@redhat.com) and +		  Peter Maydell (pmaydell@chiark.greenend.org.uk). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 of the License, or (at your option) any later version. + +   In addition to the permissions in the GNU Lesser General Public +   License, the Free Software Foundation gives you unlimited +   permission to link the compiled version of this file into +   combinations with other programs, and to distribute those +   combinations without any restriction coming from the use of this +   file.  (The Lesser General Public License restrictions do apply in +   other respects; for example, they cover modification of the file, +   and distribution when not linked into a combine executable.) + +   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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef SOFT_FP_OP_2_H +#define SOFT_FP_OP_2_H	1 + +#define _FP_FRAC_DECL_2(X)				\ +  _FP_W_TYPE X##_f0 _FP_ZERO_INIT, X##_f1 _FP_ZERO_INIT +#define _FP_FRAC_COPY_2(D, S)	(D##_f0 = S##_f0, D##_f1 = S##_f1) +#define _FP_FRAC_SET_2(X, I)	__FP_FRAC_SET_2 (X, I) +#define _FP_FRAC_HIGH_2(X)	(X##_f1) +#define _FP_FRAC_LOW_2(X)	(X##_f0) +#define _FP_FRAC_WORD_2(X, w)	(X##_f##w) + +#define _FP_FRAC_SLL_2(X, N)						\ +  (void) (((N) < _FP_W_TYPE_SIZE)					\ +	  ? ({								\ +	      if (__builtin_constant_p (N) && (N) == 1)			\ +		{							\ +		  X##_f1 = X##_f1 + X##_f1 + (((_FP_WS_TYPE) (X##_f0)) < 0); \ +		  X##_f0 += X##_f0;					\ +		}							\ +	      else							\ +		{							\ +		  X##_f1 = X##_f1 << (N) | X##_f0 >> (_FP_W_TYPE_SIZE - (N)); \ +		  X##_f0 <<= (N);					\ +		}							\ +	      0;							\ +	    })								\ +	  : ({								\ +	      X##_f1 = X##_f0 << ((N) - _FP_W_TYPE_SIZE);		\ +	      X##_f0 = 0;						\ +	    })) + + +#define _FP_FRAC_SRL_2(X, N)						\ +  (void) (((N) < _FP_W_TYPE_SIZE)					\ +	  ? ({								\ +	      X##_f0 = X##_f0 >> (N) | X##_f1 << (_FP_W_TYPE_SIZE - (N)); \ +	      X##_f1 >>= (N);						\ +	    })								\ +	  : ({								\ +	      X##_f0 = X##_f1 >> ((N) - _FP_W_TYPE_SIZE);		\ +	      X##_f1 = 0;						\ +	    })) + +/* Right shift with sticky-lsb.  */ +#define _FP_FRAC_SRST_2(X, S, N, sz)					\ +  (void) (((N) < _FP_W_TYPE_SIZE)					\ +	  ? ({								\ +	      S = (__builtin_constant_p (N) && (N) == 1			\ +		   ? X##_f0 & 1						\ +		   : (X##_f0 << (_FP_W_TYPE_SIZE - (N))) != 0);		\ +	      X##_f0 = (X##_f1 << (_FP_W_TYPE_SIZE - (N)) | X##_f0 >> (N)); \ +	      X##_f1 >>= (N);						\ +	    })								\ +	  : ({								\ +	      S = ((((N) == _FP_W_TYPE_SIZE				\ +		     ? 0						\ +		     : (X##_f1 << (2*_FP_W_TYPE_SIZE - (N))))		\ +		    | X##_f0) != 0);					\ +	      X##_f0 = (X##_f1 >> ((N) - _FP_W_TYPE_SIZE));		\ +	      X##_f1 = 0;						\ +	    })) + +#define _FP_FRAC_SRS_2(X, N, sz)					\ +  (void) (((N) < _FP_W_TYPE_SIZE)					\ +	  ? ({								\ +	      X##_f0 = (X##_f1 << (_FP_W_TYPE_SIZE - (N)) | X##_f0 >> (N) \ +			| (__builtin_constant_p (N) && (N) == 1		\ +			   ? X##_f0 & 1					\ +			   : (X##_f0 << (_FP_W_TYPE_SIZE - (N))) != 0)); \ +	      X##_f1 >>= (N);						\ +	    })								\ +	  : ({								\ +	      X##_f0 = (X##_f1 >> ((N) - _FP_W_TYPE_SIZE)		\ +			| ((((N) == _FP_W_TYPE_SIZE			\ +			     ? 0					\ +			     : (X##_f1 << (2*_FP_W_TYPE_SIZE - (N))))	\ +			    | X##_f0) != 0));				\ +	      X##_f1 = 0;						\ +	    })) + +#define _FP_FRAC_ADDI_2(X, I)	\ +  __FP_FRAC_ADDI_2 (X##_f1, X##_f0, I) + +#define _FP_FRAC_ADD_2(R, X, Y)	\ +  __FP_FRAC_ADD_2 (R##_f1, R##_f0, X##_f1, X##_f0, Y##_f1, Y##_f0) + +#define _FP_FRAC_SUB_2(R, X, Y)	\ +  __FP_FRAC_SUB_2 (R##_f1, R##_f0, X##_f1, X##_f0, Y##_f1, Y##_f0) + +#define _FP_FRAC_DEC_2(X, Y)	\ +  __FP_FRAC_DEC_2 (X##_f1, X##_f0, Y##_f1, Y##_f0) + +#define _FP_FRAC_CLZ_2(R, X)			\ +  do						\ +    {						\ +      if (X##_f1)				\ +	__FP_CLZ ((R), X##_f1);			\ +      else					\ +	{					\ +	  __FP_CLZ ((R), X##_f0);		\ +	  (R) += _FP_W_TYPE_SIZE;		\ +	}					\ +    }						\ +  while (0) + +/* Predicates.  */ +#define _FP_FRAC_NEGP_2(X)	((_FP_WS_TYPE) X##_f1 < 0) +#define _FP_FRAC_ZEROP_2(X)	((X##_f1 | X##_f0) == 0) +#define _FP_FRAC_OVERP_2(fs, X)	(_FP_FRAC_HIGH_##fs (X) & _FP_OVERFLOW_##fs) +#define _FP_FRAC_CLEAR_OVERP_2(fs, X)	(_FP_FRAC_HIGH_##fs (X) &= ~_FP_OVERFLOW_##fs) +#define _FP_FRAC_HIGHBIT_DW_2(fs, X)	\ +  (_FP_FRAC_HIGH_DW_##fs (X) & _FP_HIGHBIT_DW_##fs) +#define _FP_FRAC_EQ_2(X, Y)	(X##_f1 == Y##_f1 && X##_f0 == Y##_f0) +#define _FP_FRAC_GT_2(X, Y)	\ +  (X##_f1 > Y##_f1 || (X##_f1 == Y##_f1 && X##_f0 > Y##_f0)) +#define _FP_FRAC_GE_2(X, Y)	\ +  (X##_f1 > Y##_f1 || (X##_f1 == Y##_f1 && X##_f0 >= Y##_f0)) + +#define _FP_ZEROFRAC_2		0, 0 +#define _FP_MINFRAC_2		0, 1 +#define _FP_MAXFRAC_2		(~(_FP_WS_TYPE) 0), (~(_FP_WS_TYPE) 0) + +/* Internals.  */ + +#define __FP_FRAC_SET_2(X, I1, I0)	(X##_f0 = I0, X##_f1 = I1) + +#define __FP_CLZ_2(R, xh, xl)			\ +  do						\ +    {						\ +      if (xh)					\ +	__FP_CLZ ((R), xh);			\ +      else					\ +	{					\ +	  __FP_CLZ ((R), xl);			\ +	  (R) += _FP_W_TYPE_SIZE;		\ +	}					\ +    }						\ +  while (0) + +#if 0 + +# ifndef __FP_FRAC_ADDI_2 +#  define __FP_FRAC_ADDI_2(xh, xl, i)	\ +  (xh += ((xl += i) < i)) +# endif +# ifndef __FP_FRAC_ADD_2 +#  define __FP_FRAC_ADD_2(rh, rl, xh, xl, yh, yl)	\ +  (rh = xh + yh + ((rl = xl + yl) < xl)) +# endif +# ifndef __FP_FRAC_SUB_2 +#  define __FP_FRAC_SUB_2(rh, rl, xh, xl, yh, yl)	\ +  (rh = xh - yh - ((rl = xl - yl) > xl)) +# endif +# ifndef __FP_FRAC_DEC_2 +#  define __FP_FRAC_DEC_2(xh, xl, yh, yl)		\ +  do							\ +    {							\ +      UWtype __FP_FRAC_DEC_2_t = xl;			\ +      xh -= yh + ((xl -= yl) > __FP_FRAC_DEC_2_t);	\ +    }							\ +  while (0) +# endif + +#else + +# undef __FP_FRAC_ADDI_2 +# define __FP_FRAC_ADDI_2(xh, xl, i)	add_ssaaaa (xh, xl, xh, xl, 0, i) +# undef __FP_FRAC_ADD_2 +# define __FP_FRAC_ADD_2		add_ssaaaa +# undef __FP_FRAC_SUB_2 +# define __FP_FRAC_SUB_2		sub_ddmmss +# undef __FP_FRAC_DEC_2 +# define __FP_FRAC_DEC_2(xh, xl, yh, yl)	\ +  sub_ddmmss (xh, xl, xh, xl, yh, yl) + +#endif + +/* Unpack the raw bits of a native fp value.  Do not classify or +   normalize the data.  */ + +#define _FP_UNPACK_RAW_2(fs, X, val)			\ +  do							\ +    {							\ +      union _FP_UNION_##fs _FP_UNPACK_RAW_2_flo;	\ +      _FP_UNPACK_RAW_2_flo.flt = (val);			\ +							\ +      X##_f0 = _FP_UNPACK_RAW_2_flo.bits.frac0;		\ +      X##_f1 = _FP_UNPACK_RAW_2_flo.bits.frac1;		\ +      X##_e  = _FP_UNPACK_RAW_2_flo.bits.exp;		\ +      X##_s  = _FP_UNPACK_RAW_2_flo.bits.sign;		\ +    }							\ +  while (0) + +#define _FP_UNPACK_RAW_2_P(fs, X, val)			\ +  do							\ +    {							\ +      union _FP_UNION_##fs *_FP_UNPACK_RAW_2_P_flo	\ +	= (union _FP_UNION_##fs *) (val);		\ +							\ +      X##_f0 = _FP_UNPACK_RAW_2_P_flo->bits.frac0;	\ +      X##_f1 = _FP_UNPACK_RAW_2_P_flo->bits.frac1;	\ +      X##_e  = _FP_UNPACK_RAW_2_P_flo->bits.exp;	\ +      X##_s  = _FP_UNPACK_RAW_2_P_flo->bits.sign;	\ +    }							\ +  while (0) + + +/* Repack the raw bits of a native fp value.  */ + +#define _FP_PACK_RAW_2(fs, val, X)		\ +  do						\ +    {						\ +      union _FP_UNION_##fs _FP_PACK_RAW_2_flo;	\ +						\ +      _FP_PACK_RAW_2_flo.bits.frac0 = X##_f0;	\ +      _FP_PACK_RAW_2_flo.bits.frac1 = X##_f1;	\ +      _FP_PACK_RAW_2_flo.bits.exp   = X##_e;	\ +      _FP_PACK_RAW_2_flo.bits.sign  = X##_s;	\ +						\ +      (val) = _FP_PACK_RAW_2_flo.flt;		\ +    }						\ +  while (0) + +#define _FP_PACK_RAW_2_P(fs, val, X)			\ +  do							\ +    {							\ +      union _FP_UNION_##fs *_FP_PACK_RAW_2_P_flo	\ +	= (union _FP_UNION_##fs *) (val);		\ +							\ +      _FP_PACK_RAW_2_P_flo->bits.frac0 = X##_f0;	\ +      _FP_PACK_RAW_2_P_flo->bits.frac1 = X##_f1;	\ +      _FP_PACK_RAW_2_P_flo->bits.exp   = X##_e;		\ +      _FP_PACK_RAW_2_P_flo->bits.sign  = X##_s;		\ +    }							\ +  while (0) + + +/* Multiplication algorithms: */ + +/* Given a 1W * 1W => 2W primitive, do the extended multiplication.  */ + +#define _FP_MUL_MEAT_DW_2_wide(wfracbits, R, X, Y, doit)		\ +  do									\ +    {									\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_DW_2_wide_b);			\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_DW_2_wide_c);			\ +									\ +      doit (_FP_FRAC_WORD_4 (R, 1), _FP_FRAC_WORD_4 (R, 0),		\ +	    X##_f0, Y##_f0);						\ +      doit (_FP_MUL_MEAT_DW_2_wide_b_f1, _FP_MUL_MEAT_DW_2_wide_b_f0,	\ +	    X##_f0, Y##_f1);						\ +      doit (_FP_MUL_MEAT_DW_2_wide_c_f1, _FP_MUL_MEAT_DW_2_wide_c_f0,	\ +	    X##_f1, Y##_f0);						\ +      doit (_FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2),		\ +	    X##_f1, Y##_f1);						\ +									\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2),	\ +		       _FP_FRAC_WORD_4 (R, 1), 0,			\ +		       _FP_MUL_MEAT_DW_2_wide_b_f1,			\ +		       _FP_MUL_MEAT_DW_2_wide_b_f0,			\ +		       _FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2),	\ +		       _FP_FRAC_WORD_4 (R, 1));				\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2),	\ +		       _FP_FRAC_WORD_4 (R, 1), 0,			\ +		       _FP_MUL_MEAT_DW_2_wide_c_f1,			\ +		       _FP_MUL_MEAT_DW_2_wide_c_f0,			\ +		       _FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2),	\ +		       _FP_FRAC_WORD_4 (R, 1));				\ +    }									\ +  while (0) + +#define _FP_MUL_MEAT_2_wide(wfracbits, R, X, Y, doit)			\ +  do									\ +    {									\ +      _FP_FRAC_DECL_4 (_FP_MUL_MEAT_2_wide_z);				\ +									\ +      _FP_MUL_MEAT_DW_2_wide ((wfracbits), _FP_MUL_MEAT_2_wide_z,	\ +			      X, Y, doit);				\ +									\ +      /* Normalize since we know where the msb of the multiplicands	\ +	 were (bit B), we know that the msb of the of the product is	\ +	 at either 2B or 2B-1.  */					\ +      _FP_FRAC_SRS_4 (_FP_MUL_MEAT_2_wide_z, (wfracbits)-1,		\ +		      2*(wfracbits));					\ +      R##_f0 = _FP_FRAC_WORD_4 (_FP_MUL_MEAT_2_wide_z, 0);		\ +      R##_f1 = _FP_FRAC_WORD_4 (_FP_MUL_MEAT_2_wide_z, 1);		\ +    }									\ +  while (0) + +/* Given a 1W * 1W => 2W primitive, do the extended multiplication. +   Do only 3 multiplications instead of four. This one is for machines +   where multiplication is much more expensive than subtraction.  */ + +#define _FP_MUL_MEAT_DW_2_wide_3mul(wfracbits, R, X, Y, doit)		\ +  do									\ +    {									\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_DW_2_wide_3mul_b);			\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_DW_2_wide_3mul_c);			\ +      _FP_W_TYPE _FP_MUL_MEAT_DW_2_wide_3mul_d;				\ +      int _FP_MUL_MEAT_DW_2_wide_3mul_c1;				\ +      int _FP_MUL_MEAT_DW_2_wide_3mul_c2;				\ +									\ +      _FP_MUL_MEAT_DW_2_wide_3mul_b_f0 = X##_f0 + X##_f1;		\ +      _FP_MUL_MEAT_DW_2_wide_3mul_c1					\ +	= _FP_MUL_MEAT_DW_2_wide_3mul_b_f0 < X##_f0;			\ +      _FP_MUL_MEAT_DW_2_wide_3mul_b_f1 = Y##_f0 + Y##_f1;		\ +      _FP_MUL_MEAT_DW_2_wide_3mul_c2					\ +	= _FP_MUL_MEAT_DW_2_wide_3mul_b_f1 < Y##_f0;			\ +      doit (_FP_MUL_MEAT_DW_2_wide_3mul_d, _FP_FRAC_WORD_4 (R, 0),	\ +	    X##_f0, Y##_f0);						\ +      doit (_FP_FRAC_WORD_4 (R, 2), _FP_FRAC_WORD_4 (R, 1),		\ +	    _FP_MUL_MEAT_DW_2_wide_3mul_b_f0,				\ +	    _FP_MUL_MEAT_DW_2_wide_3mul_b_f1);				\ +      doit (_FP_MUL_MEAT_DW_2_wide_3mul_c_f1,				\ +	    _FP_MUL_MEAT_DW_2_wide_3mul_c_f0, X##_f1, Y##_f1);		\ +									\ +      _FP_MUL_MEAT_DW_2_wide_3mul_b_f0					\ +	&= -_FP_MUL_MEAT_DW_2_wide_3mul_c2;				\ +      _FP_MUL_MEAT_DW_2_wide_3mul_b_f1					\ +	&= -_FP_MUL_MEAT_DW_2_wide_3mul_c1;				\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2),	\ +		       _FP_FRAC_WORD_4 (R, 1),				\ +		       (_FP_MUL_MEAT_DW_2_wide_3mul_c1			\ +			& _FP_MUL_MEAT_DW_2_wide_3mul_c2), 0,		\ +		       _FP_MUL_MEAT_DW_2_wide_3mul_d,			\ +		       0, _FP_FRAC_WORD_4 (R, 2), _FP_FRAC_WORD_4 (R, 1)); \ +      __FP_FRAC_ADDI_2 (_FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2),	\ +			_FP_MUL_MEAT_DW_2_wide_3mul_b_f0);		\ +      __FP_FRAC_ADDI_2 (_FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2),	\ +			_FP_MUL_MEAT_DW_2_wide_3mul_b_f1);		\ +      __FP_FRAC_DEC_3 (_FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2),	\ +		       _FP_FRAC_WORD_4 (R, 1),				\ +		       0, _FP_MUL_MEAT_DW_2_wide_3mul_d,		\ +		       _FP_FRAC_WORD_4 (R, 0));				\ +      __FP_FRAC_DEC_3 (_FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2),	\ +		       _FP_FRAC_WORD_4 (R, 1), 0,			\ +		       _FP_MUL_MEAT_DW_2_wide_3mul_c_f1,		\ +		       _FP_MUL_MEAT_DW_2_wide_3mul_c_f0);		\ +      __FP_FRAC_ADD_2 (_FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2),	\ +		       _FP_MUL_MEAT_DW_2_wide_3mul_c_f1,		\ +		       _FP_MUL_MEAT_DW_2_wide_3mul_c_f0,		\ +		       _FP_FRAC_WORD_4 (R, 3), _FP_FRAC_WORD_4 (R, 2));	\ +    }									\ +  while (0) + +#define _FP_MUL_MEAT_2_wide_3mul(wfracbits, R, X, Y, doit)		\ +  do									\ +    {									\ +      _FP_FRAC_DECL_4 (_FP_MUL_MEAT_2_wide_3mul_z);			\ +									\ +      _FP_MUL_MEAT_DW_2_wide_3mul ((wfracbits),				\ +				   _FP_MUL_MEAT_2_wide_3mul_z,		\ +				   X, Y, doit);				\ +									\ +      /* Normalize since we know where the msb of the multiplicands	\ +	 were (bit B), we know that the msb of the of the product is	\ +	 at either 2B or 2B-1.  */					\ +      _FP_FRAC_SRS_4 (_FP_MUL_MEAT_2_wide_3mul_z,			\ +		      (wfracbits)-1, 2*(wfracbits));			\ +      R##_f0 = _FP_FRAC_WORD_4 (_FP_MUL_MEAT_2_wide_3mul_z, 0);		\ +      R##_f1 = _FP_FRAC_WORD_4 (_FP_MUL_MEAT_2_wide_3mul_z, 1);		\ +    }									\ +  while (0) + +#define _FP_MUL_MEAT_DW_2_gmp(wfracbits, R, X, Y)	\ +  do							\ +    {							\ +      _FP_W_TYPE _FP_MUL_MEAT_DW_2_gmp_x[2];		\ +      _FP_W_TYPE _FP_MUL_MEAT_DW_2_gmp_y[2];		\ +      _FP_MUL_MEAT_DW_2_gmp_x[0] = X##_f0;		\ +      _FP_MUL_MEAT_DW_2_gmp_x[1] = X##_f1;		\ +      _FP_MUL_MEAT_DW_2_gmp_y[0] = Y##_f0;		\ +      _FP_MUL_MEAT_DW_2_gmp_y[1] = Y##_f1;		\ +							\ +      mpn_mul_n (R##_f, _FP_MUL_MEAT_DW_2_gmp_x,	\ +		 _FP_MUL_MEAT_DW_2_gmp_y, 2);		\ +    }							\ +  while (0) + +#define _FP_MUL_MEAT_2_gmp(wfracbits, R, X, Y)				\ +  do									\ +    {									\ +      _FP_FRAC_DECL_4 (_FP_MUL_MEAT_2_gmp_z);				\ +									\ +      _FP_MUL_MEAT_DW_2_gmp ((wfracbits), _FP_MUL_MEAT_2_gmp_z, X, Y);	\ +									\ +      /* Normalize since we know where the msb of the multiplicands	\ +	 were (bit B), we know that the msb of the of the product is	\ +	 at either 2B or 2B-1.  */					\ +      _FP_FRAC_SRS_4 (_FP_MUL_MEAT_2_gmp_z, (wfracbits)-1,		\ +		      2*(wfracbits));					\ +      R##_f0 = _FP_MUL_MEAT_2_gmp_z_f[0];				\ +      R##_f1 = _FP_MUL_MEAT_2_gmp_z_f[1];				\ +    }									\ +  while (0) + +/* Do at most 120x120=240 bits multiplication using double floating +   point multiplication.  This is useful if floating point +   multiplication has much bigger throughput than integer multiply. +   It is supposed to work for _FP_W_TYPE_SIZE 64 and wfracbits +   between 106 and 120 only. +   Caller guarantees that X and Y has (1LLL << (wfracbits - 1)) set. +   SETFETZ is a macro which will disable all FPU exceptions and set rounding +   towards zero,  RESETFE should optionally reset it back.  */ + +#define _FP_MUL_MEAT_2_120_240_double(wfracbits, R, X, Y, setfetz, resetfe) \ +  do									\ +    {									\ +      static const double _const[] =					\ +	{								\ +	  /* 2^-24 */ 5.9604644775390625e-08,				\ +	  /* 2^-48 */ 3.5527136788005009e-15,				\ +	  /* 2^-72 */ 2.1175823681357508e-22,				\ +	  /* 2^-96 */ 1.2621774483536189e-29,				\ +	  /* 2^28 */ 2.68435456e+08,					\ +	  /* 2^4 */ 1.600000e+01,					\ +	  /* 2^-20 */ 9.5367431640625e-07,				\ +	  /* 2^-44 */ 5.6843418860808015e-14,				\ +	  /* 2^-68 */ 3.3881317890172014e-21,				\ +	  /* 2^-92 */ 2.0194839173657902e-28,				\ +	  /* 2^-116 */ 1.2037062152420224e-35				\ +	};								\ +      double _a240, _b240, _c240, _d240, _e240, _f240,			\ +	_g240, _h240, _i240, _j240, _k240;				\ +      union { double d; UDItype i; } _l240, _m240, _n240, _o240,	\ +				       _p240, _q240, _r240, _s240;	\ +      UDItype _t240, _u240, _v240, _w240, _x240, _y240 = 0;		\ +									\ +      _FP_STATIC_ASSERT ((wfracbits) >= 106 && (wfracbits) <= 120,	\ +			 "wfracbits out of range");			\ +									\ +      setfetz;								\ +									\ +      _e240 = (double) (long) (X##_f0 & 0xffffff);			\ +      _j240 = (double) (long) (Y##_f0 & 0xffffff);			\ +      _d240 = (double) (long) ((X##_f0 >> 24) & 0xffffff);		\ +      _i240 = (double) (long) ((Y##_f0 >> 24) & 0xffffff);		\ +      _c240 = (double) (long) (((X##_f1 << 16) & 0xffffff) | (X##_f0 >> 48)); \ +      _h240 = (double) (long) (((Y##_f1 << 16) & 0xffffff) | (Y##_f0 >> 48)); \ +      _b240 = (double) (long) ((X##_f1 >> 8) & 0xffffff);		\ +      _g240 = (double) (long) ((Y##_f1 >> 8) & 0xffffff);		\ +      _a240 = (double) (long) (X##_f1 >> 32);				\ +      _f240 = (double) (long) (Y##_f1 >> 32);				\ +      _e240 *= _const[3];						\ +      _j240 *= _const[3];						\ +      _d240 *= _const[2];						\ +      _i240 *= _const[2];						\ +      _c240 *= _const[1];						\ +      _h240 *= _const[1];						\ +      _b240 *= _const[0];						\ +      _g240 *= _const[0];						\ +      _s240.d =							      _e240*_j240; \ +      _r240.d =						_d240*_j240 + _e240*_i240; \ +      _q240.d =				  _c240*_j240 + _d240*_i240 + _e240*_h240; \ +      _p240.d =		    _b240*_j240 + _c240*_i240 + _d240*_h240 + _e240*_g240; \ +      _o240.d = _a240*_j240 + _b240*_i240 + _c240*_h240 + _d240*_g240 + _e240*_f240; \ +      _n240.d = _a240*_i240 + _b240*_h240 + _c240*_g240 + _d240*_f240;	\ +      _m240.d = _a240*_h240 + _b240*_g240 + _c240*_f240;		\ +      _l240.d = _a240*_g240 + _b240*_f240;				\ +      _k240 =   _a240*_f240;						\ +      _r240.d += _s240.d;						\ +      _q240.d += _r240.d;						\ +      _p240.d += _q240.d;						\ +      _o240.d += _p240.d;						\ +      _n240.d += _o240.d;						\ +      _m240.d += _n240.d;						\ +      _l240.d += _m240.d;						\ +      _k240 += _l240.d;							\ +      _s240.d -= ((_const[10]+_s240.d)-_const[10]);			\ +      _r240.d -= ((_const[9]+_r240.d)-_const[9]);			\ +      _q240.d -= ((_const[8]+_q240.d)-_const[8]);			\ +      _p240.d -= ((_const[7]+_p240.d)-_const[7]);			\ +      _o240.d += _const[7];						\ +      _n240.d += _const[6];						\ +      _m240.d += _const[5];						\ +      _l240.d += _const[4];						\ +      if (_s240.d != 0.0)						\ +	_y240 = 1;							\ +      if (_r240.d != 0.0)						\ +	_y240 = 1;							\ +      if (_q240.d != 0.0)						\ +	_y240 = 1;							\ +      if (_p240.d != 0.0)						\ +	_y240 = 1;							\ +      _t240 = (DItype) _k240;						\ +      _u240 = _l240.i;							\ +      _v240 = _m240.i;							\ +      _w240 = _n240.i;							\ +      _x240 = _o240.i;							\ +      R##_f1 = ((_t240 << (128 - (wfracbits - 1)))			\ +		| ((_u240 & 0xffffff) >> ((wfracbits - 1) - 104)));	\ +      R##_f0 = (((_u240 & 0xffffff) << (168 - (wfracbits - 1)))		\ +		| ((_v240 & 0xffffff) << (144 - (wfracbits - 1)))	\ +		| ((_w240 & 0xffffff) << (120 - (wfracbits - 1)))	\ +		| ((_x240 & 0xffffff) >> ((wfracbits - 1) - 96))	\ +		| _y240);						\ +      resetfe;								\ +    }									\ +  while (0) + +/* Division algorithms: */ + +#define _FP_DIV_MEAT_2_udiv(fs, R, X, Y)				\ +  do									\ +    {									\ +      _FP_W_TYPE _FP_DIV_MEAT_2_udiv_n_f2;				\ +      _FP_W_TYPE _FP_DIV_MEAT_2_udiv_n_f1;				\ +      _FP_W_TYPE _FP_DIV_MEAT_2_udiv_n_f0;				\ +      _FP_W_TYPE _FP_DIV_MEAT_2_udiv_r_f1;				\ +      _FP_W_TYPE _FP_DIV_MEAT_2_udiv_r_f0;				\ +      _FP_W_TYPE _FP_DIV_MEAT_2_udiv_m_f1;				\ +      _FP_W_TYPE _FP_DIV_MEAT_2_udiv_m_f0;				\ +      if (_FP_FRAC_GE_2 (X, Y))						\ +	{								\ +	  _FP_DIV_MEAT_2_udiv_n_f2 = X##_f1 >> 1;			\ +	  _FP_DIV_MEAT_2_udiv_n_f1					\ +	    = X##_f1 << (_FP_W_TYPE_SIZE - 1) | X##_f0 >> 1;		\ +	  _FP_DIV_MEAT_2_udiv_n_f0					\ +	    = X##_f0 << (_FP_W_TYPE_SIZE - 1);				\ +	}								\ +      else								\ +	{								\ +	  R##_e--;							\ +	  _FP_DIV_MEAT_2_udiv_n_f2 = X##_f1;				\ +	  _FP_DIV_MEAT_2_udiv_n_f1 = X##_f0;				\ +	  _FP_DIV_MEAT_2_udiv_n_f0 = 0;					\ +	}								\ +									\ +      /* Normalize, i.e. make the most significant bit of the		\ +	 denominator set.  */						\ +      _FP_FRAC_SLL_2 (Y, _FP_WFRACXBITS_##fs);				\ +									\ +      udiv_qrnnd (R##_f1, _FP_DIV_MEAT_2_udiv_r_f1,			\ +		  _FP_DIV_MEAT_2_udiv_n_f2, _FP_DIV_MEAT_2_udiv_n_f1,	\ +		  Y##_f1);						\ +      umul_ppmm (_FP_DIV_MEAT_2_udiv_m_f1, _FP_DIV_MEAT_2_udiv_m_f0,	\ +		 R##_f1, Y##_f0);					\ +      _FP_DIV_MEAT_2_udiv_r_f0 = _FP_DIV_MEAT_2_udiv_n_f0;		\ +      if (_FP_FRAC_GT_2 (_FP_DIV_MEAT_2_udiv_m, _FP_DIV_MEAT_2_udiv_r))	\ +	{								\ +	  R##_f1--;							\ +	  _FP_FRAC_ADD_2 (_FP_DIV_MEAT_2_udiv_r, Y,			\ +			  _FP_DIV_MEAT_2_udiv_r);			\ +	  if (_FP_FRAC_GE_2 (_FP_DIV_MEAT_2_udiv_r, Y)			\ +	      && _FP_FRAC_GT_2 (_FP_DIV_MEAT_2_udiv_m,			\ +				_FP_DIV_MEAT_2_udiv_r))			\ +	    {								\ +	      R##_f1--;							\ +	      _FP_FRAC_ADD_2 (_FP_DIV_MEAT_2_udiv_r, Y,			\ +			      _FP_DIV_MEAT_2_udiv_r);			\ +	    }								\ +	}								\ +      _FP_FRAC_DEC_2 (_FP_DIV_MEAT_2_udiv_r, _FP_DIV_MEAT_2_udiv_m);	\ +									\ +      if (_FP_DIV_MEAT_2_udiv_r_f1 == Y##_f1)				\ +	{								\ +	  /* This is a special case, not an optimization		\ +	     (_FP_DIV_MEAT_2_udiv_r/Y##_f1 would not fit into UWtype).	\ +	     As _FP_DIV_MEAT_2_udiv_r is guaranteed to be < Y,		\ +	     R##_f0 can be either (UWtype)-1 or (UWtype)-2.  But as we	\ +	     know what kind of bits it is (sticky, guard, round),	\ +	     we don't care.  We also don't care what the reminder is,	\ +	     because the guard bit will be set anyway.  -jj */		\ +	  R##_f0 = -1;							\ +	}								\ +      else								\ +	{								\ +	  udiv_qrnnd (R##_f0, _FP_DIV_MEAT_2_udiv_r_f1,			\ +		      _FP_DIV_MEAT_2_udiv_r_f1,				\ +		      _FP_DIV_MEAT_2_udiv_r_f0, Y##_f1);		\ +	  umul_ppmm (_FP_DIV_MEAT_2_udiv_m_f1,				\ +		     _FP_DIV_MEAT_2_udiv_m_f0, R##_f0, Y##_f0);		\ +	  _FP_DIV_MEAT_2_udiv_r_f0 = 0;					\ +	  if (_FP_FRAC_GT_2 (_FP_DIV_MEAT_2_udiv_m,			\ +			     _FP_DIV_MEAT_2_udiv_r))			\ +	    {								\ +	      R##_f0--;							\ +	      _FP_FRAC_ADD_2 (_FP_DIV_MEAT_2_udiv_r, Y,			\ +			      _FP_DIV_MEAT_2_udiv_r);			\ +	      if (_FP_FRAC_GE_2 (_FP_DIV_MEAT_2_udiv_r, Y)		\ +		  && _FP_FRAC_GT_2 (_FP_DIV_MEAT_2_udiv_m,		\ +				    _FP_DIV_MEAT_2_udiv_r))		\ +		{							\ +		  R##_f0--;						\ +		  _FP_FRAC_ADD_2 (_FP_DIV_MEAT_2_udiv_r, Y,		\ +				  _FP_DIV_MEAT_2_udiv_r);		\ +		}							\ +	    }								\ +	  if (!_FP_FRAC_EQ_2 (_FP_DIV_MEAT_2_udiv_r,			\ +			      _FP_DIV_MEAT_2_udiv_m))			\ +	    R##_f0 |= _FP_WORK_STICKY;					\ +	}								\ +    }									\ +  while (0) + + +/* Square root algorithms: +   We have just one right now, maybe Newton approximation +   should be added for those machines where division is fast.  */ + +#define _FP_SQRT_MEAT_2(R, S, T, X, q)				\ +  do								\ +    {								\ +      while (q)							\ +	{							\ +	  T##_f1 = S##_f1 + (q);				\ +	  if (T##_f1 <= X##_f1)					\ +	    {							\ +	      S##_f1 = T##_f1 + (q);				\ +	      X##_f1 -= T##_f1;					\ +	      R##_f1 += (q);					\ +	    }							\ +	  _FP_FRAC_SLL_2 (X, 1);				\ +	  (q) >>= 1;						\ +	}							\ +      (q) = (_FP_W_TYPE) 1 << (_FP_W_TYPE_SIZE - 1);		\ +      while ((q) != _FP_WORK_ROUND)				\ +	{							\ +	  T##_f0 = S##_f0 + (q);				\ +	  T##_f1 = S##_f1;					\ +	  if (T##_f1 < X##_f1					\ +	      || (T##_f1 == X##_f1 && T##_f0 <= X##_f0))	\ +	    {							\ +	      S##_f0 = T##_f0 + (q);				\ +	      S##_f1 += (T##_f0 > S##_f0);			\ +	      _FP_FRAC_DEC_2 (X, T);				\ +	      R##_f0 += (q);					\ +	    }							\ +	  _FP_FRAC_SLL_2 (X, 1);				\ +	  (q) >>= 1;						\ +	}							\ +      if (X##_f0 | X##_f1)					\ +	{							\ +	  if (S##_f1 < X##_f1					\ +	      || (S##_f1 == X##_f1 && S##_f0 < X##_f0))		\ +	    R##_f0 |= _FP_WORK_ROUND;				\ +	  R##_f0 |= _FP_WORK_STICKY;				\ +	}							\ +    }								\ +  while (0) + + +/* Assembly/disassembly for converting to/from integral types. +   No shifting or overflow handled here.  */ + +#define _FP_FRAC_ASSEMBLE_2(r, X, rsize)	\ +  (void) (((rsize) <= _FP_W_TYPE_SIZE)		\ +	  ? ({ (r) = X##_f0; })			\ +	  : ({					\ +	      (r) = X##_f1;			\ +	      (r) <<= _FP_W_TYPE_SIZE;		\ +	      (r) += X##_f0;			\ +	    })) + +#define _FP_FRAC_DISASSEMBLE_2(X, r, rsize)	\ +  do						\ +    {						\ +      X##_f0 = (r);				\ +      X##_f1 = ((rsize) <= _FP_W_TYPE_SIZE	\ +		? 0				\ +		: (r) >> _FP_W_TYPE_SIZE);	\ +    }						\ +  while (0) + +/* Convert FP values between word sizes.  */ + +#define _FP_FRAC_COPY_1_2(D, S)		(D##_f = S##_f0) + +#define _FP_FRAC_COPY_2_1(D, S)		((D##_f0 = S##_f), (D##_f1 = 0)) + +#define _FP_FRAC_COPY_2_2(D, S)		_FP_FRAC_COPY_2 (D, S) + +#endif /* !SOFT_FP_OP_2_H */ diff --git a/libc/sysdeps/linux/sparc64/soft-fp/op-4.h b/libc/sysdeps/linux/sparc64/soft-fp/op-4.h new file mode 100644 index 000000000..5034667e3 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/op-4.h @@ -0,0 +1,875 @@ +/* Software floating-point emulation. +   Basic four-word fraction declaration and manipulation. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com), +		  Jakub Jelinek (jj@ultra.linux.cz), +		  David S. Miller (davem@redhat.com) and +		  Peter Maydell (pmaydell@chiark.greenend.org.uk). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 of the License, or (at your option) any later version. + +   In addition to the permissions in the GNU Lesser General Public +   License, the Free Software Foundation gives you unlimited +   permission to link the compiled version of this file into +   combinations with other programs, and to distribute those +   combinations without any restriction coming from the use of this +   file.  (The Lesser General Public License restrictions do apply in +   other respects; for example, they cover modification of the file, +   and distribution when not linked into a combine executable.) + +   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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef SOFT_FP_OP_4_H +#define SOFT_FP_OP_4_H	1 + +#define _FP_FRAC_DECL_4(X)	_FP_W_TYPE X##_f[4] +#define _FP_FRAC_COPY_4(D, S)			\ +  (D##_f[0] = S##_f[0], D##_f[1] = S##_f[1],	\ +   D##_f[2] = S##_f[2], D##_f[3] = S##_f[3]) +#define _FP_FRAC_SET_4(X, I)	__FP_FRAC_SET_4 (X, I) +#define _FP_FRAC_HIGH_4(X)	(X##_f[3]) +#define _FP_FRAC_LOW_4(X)	(X##_f[0]) +#define _FP_FRAC_WORD_4(X, w)	(X##_f[w]) + +#define _FP_FRAC_SLL_4(X, N)						\ +  do									\ +    {									\ +      _FP_I_TYPE _FP_FRAC_SLL_4_up, _FP_FRAC_SLL_4_down;		\ +      _FP_I_TYPE _FP_FRAC_SLL_4_skip, _FP_FRAC_SLL_4_i;			\ +      _FP_FRAC_SLL_4_skip = (N) / _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SLL_4_up = (N) % _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SLL_4_down = _FP_W_TYPE_SIZE - _FP_FRAC_SLL_4_up;	\ +      if (!_FP_FRAC_SLL_4_up)						\ +	for (_FP_FRAC_SLL_4_i = 3;					\ +	     _FP_FRAC_SLL_4_i >= _FP_FRAC_SLL_4_skip;			\ +	     --_FP_FRAC_SLL_4_i)					\ +	  X##_f[_FP_FRAC_SLL_4_i]					\ +	    = X##_f[_FP_FRAC_SLL_4_i-_FP_FRAC_SLL_4_skip];		\ +      else								\ +	{								\ +	  for (_FP_FRAC_SLL_4_i = 3;					\ +	       _FP_FRAC_SLL_4_i > _FP_FRAC_SLL_4_skip;			\ +	       --_FP_FRAC_SLL_4_i)					\ +	    X##_f[_FP_FRAC_SLL_4_i]					\ +	      = ((X##_f[_FP_FRAC_SLL_4_i-_FP_FRAC_SLL_4_skip]		\ +		  << _FP_FRAC_SLL_4_up)					\ +		 | (X##_f[_FP_FRAC_SLL_4_i-_FP_FRAC_SLL_4_skip-1]	\ +		    >> _FP_FRAC_SLL_4_down));				\ +	  X##_f[_FP_FRAC_SLL_4_i--] = X##_f[0] << _FP_FRAC_SLL_4_up;	\ +	}								\ +      for (; _FP_FRAC_SLL_4_i >= 0; --_FP_FRAC_SLL_4_i)			\ +	X##_f[_FP_FRAC_SLL_4_i] = 0;					\ +    }									\ +  while (0) + +/* This one was broken too.  */ +#define _FP_FRAC_SRL_4(X, N)						\ +  do									\ +    {									\ +      _FP_I_TYPE _FP_FRAC_SRL_4_up, _FP_FRAC_SRL_4_down;		\ +      _FP_I_TYPE _FP_FRAC_SRL_4_skip, _FP_FRAC_SRL_4_i;			\ +      _FP_FRAC_SRL_4_skip = (N) / _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SRL_4_down = (N) % _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SRL_4_up = _FP_W_TYPE_SIZE - _FP_FRAC_SRL_4_down;	\ +      if (!_FP_FRAC_SRL_4_down)						\ +	for (_FP_FRAC_SRL_4_i = 0;					\ +	     _FP_FRAC_SRL_4_i <= 3-_FP_FRAC_SRL_4_skip;			\ +	     ++_FP_FRAC_SRL_4_i)					\ +	  X##_f[_FP_FRAC_SRL_4_i]					\ +	    = X##_f[_FP_FRAC_SRL_4_i+_FP_FRAC_SRL_4_skip];		\ +      else								\ +	{								\ +	  for (_FP_FRAC_SRL_4_i = 0;					\ +	       _FP_FRAC_SRL_4_i < 3-_FP_FRAC_SRL_4_skip;		\ +	       ++_FP_FRAC_SRL_4_i)					\ +	    X##_f[_FP_FRAC_SRL_4_i]					\ +	      = ((X##_f[_FP_FRAC_SRL_4_i+_FP_FRAC_SRL_4_skip]		\ +		  >> _FP_FRAC_SRL_4_down)				\ +		 | (X##_f[_FP_FRAC_SRL_4_i+_FP_FRAC_SRL_4_skip+1]	\ +		    << _FP_FRAC_SRL_4_up));				\ +	  X##_f[_FP_FRAC_SRL_4_i++] = X##_f[3] >> _FP_FRAC_SRL_4_down;	\ +	}								\ +      for (; _FP_FRAC_SRL_4_i < 4; ++_FP_FRAC_SRL_4_i)			\ +	X##_f[_FP_FRAC_SRL_4_i] = 0;					\ +    }									\ +  while (0) + + +/* Right shift with sticky-lsb. +   What this actually means is that we do a standard right-shift, +   but that if any of the bits that fall off the right hand side +   were one then we always set the LSbit.  */ +#define _FP_FRAC_SRST_4(X, S, N, size)					\ +  do									\ +    {									\ +      _FP_I_TYPE _FP_FRAC_SRST_4_up, _FP_FRAC_SRST_4_down;		\ +      _FP_I_TYPE _FP_FRAC_SRST_4_skip, _FP_FRAC_SRST_4_i;		\ +      _FP_W_TYPE _FP_FRAC_SRST_4_s;					\ +      _FP_FRAC_SRST_4_skip = (N) / _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SRST_4_down = (N) % _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SRST_4_up = _FP_W_TYPE_SIZE - _FP_FRAC_SRST_4_down;	\ +      for (_FP_FRAC_SRST_4_s = _FP_FRAC_SRST_4_i = 0;			\ +	   _FP_FRAC_SRST_4_i < _FP_FRAC_SRST_4_skip;			\ +	   ++_FP_FRAC_SRST_4_i)						\ +	_FP_FRAC_SRST_4_s |= X##_f[_FP_FRAC_SRST_4_i];			\ +      if (!_FP_FRAC_SRST_4_down)					\ +	for (_FP_FRAC_SRST_4_i = 0;					\ +	     _FP_FRAC_SRST_4_i <= 3-_FP_FRAC_SRST_4_skip;		\ +	     ++_FP_FRAC_SRST_4_i)					\ +	  X##_f[_FP_FRAC_SRST_4_i]					\ +	    = X##_f[_FP_FRAC_SRST_4_i+_FP_FRAC_SRST_4_skip];		\ +      else								\ +	{								\ +	  _FP_FRAC_SRST_4_s						\ +	    |= X##_f[_FP_FRAC_SRST_4_i] << _FP_FRAC_SRST_4_up;		\ +	  for (_FP_FRAC_SRST_4_i = 0;					\ +	       _FP_FRAC_SRST_4_i < 3-_FP_FRAC_SRST_4_skip;		\ +	       ++_FP_FRAC_SRST_4_i)					\ +	    X##_f[_FP_FRAC_SRST_4_i]					\ +	      = ((X##_f[_FP_FRAC_SRST_4_i+_FP_FRAC_SRST_4_skip]		\ +		  >> _FP_FRAC_SRST_4_down)				\ +		 | (X##_f[_FP_FRAC_SRST_4_i+_FP_FRAC_SRST_4_skip+1]	\ +		    << _FP_FRAC_SRST_4_up));				\ +	  X##_f[_FP_FRAC_SRST_4_i++]					\ +	    = X##_f[3] >> _FP_FRAC_SRST_4_down;				\ +	}								\ +      for (; _FP_FRAC_SRST_4_i < 4; ++_FP_FRAC_SRST_4_i)		\ +	X##_f[_FP_FRAC_SRST_4_i] = 0;					\ +      S = (_FP_FRAC_SRST_4_s != 0);					\ +    }									\ +  while (0) + +#define _FP_FRAC_SRS_4(X, N, size)				\ +  do								\ +    {								\ +      int _FP_FRAC_SRS_4_sticky;				\ +      _FP_FRAC_SRST_4 (X, _FP_FRAC_SRS_4_sticky, (N), (size));	\ +      X##_f[0] |= _FP_FRAC_SRS_4_sticky;			\ +    }								\ +  while (0) + +#define _FP_FRAC_ADD_4(R, X, Y)					\ +  __FP_FRAC_ADD_4 (R##_f[3], R##_f[2], R##_f[1], R##_f[0],	\ +		   X##_f[3], X##_f[2], X##_f[1], X##_f[0],	\ +		   Y##_f[3], Y##_f[2], Y##_f[1], Y##_f[0]) + +#define _FP_FRAC_SUB_4(R, X, Y)					\ +  __FP_FRAC_SUB_4 (R##_f[3], R##_f[2], R##_f[1], R##_f[0],	\ +		   X##_f[3], X##_f[2], X##_f[1], X##_f[0],	\ +		   Y##_f[3], Y##_f[2], Y##_f[1], Y##_f[0]) + +#define _FP_FRAC_DEC_4(X, Y)					\ +  __FP_FRAC_DEC_4 (X##_f[3], X##_f[2], X##_f[1], X##_f[0],	\ +		   Y##_f[3], Y##_f[2], Y##_f[1], Y##_f[0]) + +#define _FP_FRAC_ADDI_4(X, I)					\ +  __FP_FRAC_ADDI_4 (X##_f[3], X##_f[2], X##_f[1], X##_f[0], I) + +#define _FP_ZEROFRAC_4  0, 0, 0, 0 +#define _FP_MINFRAC_4   0, 0, 0, 1 +#define _FP_MAXFRAC_4	(~(_FP_WS_TYPE) 0), (~(_FP_WS_TYPE) 0), (~(_FP_WS_TYPE) 0), (~(_FP_WS_TYPE) 0) + +#define _FP_FRAC_ZEROP_4(X)     ((X##_f[0] | X##_f[1] | X##_f[2] | X##_f[3]) == 0) +#define _FP_FRAC_NEGP_4(X)      ((_FP_WS_TYPE) X##_f[3] < 0) +#define _FP_FRAC_OVERP_4(fs, X)  (_FP_FRAC_HIGH_##fs (X) & _FP_OVERFLOW_##fs) +#define _FP_FRAC_HIGHBIT_DW_4(fs, X)	\ +  (_FP_FRAC_HIGH_DW_##fs (X) & _FP_HIGHBIT_DW_##fs) +#define _FP_FRAC_CLEAR_OVERP_4(fs, X)  (_FP_FRAC_HIGH_##fs (X) &= ~_FP_OVERFLOW_##fs) + +#define _FP_FRAC_EQ_4(X, Y)				\ +  (X##_f[0] == Y##_f[0] && X##_f[1] == Y##_f[1]		\ +   && X##_f[2] == Y##_f[2] && X##_f[3] == Y##_f[3]) + +#define _FP_FRAC_GT_4(X, Y)				\ +  (X##_f[3] > Y##_f[3]					\ +   || (X##_f[3] == Y##_f[3]				\ +       && (X##_f[2] > Y##_f[2]				\ +	   || (X##_f[2] == Y##_f[2]			\ +	       && (X##_f[1] > Y##_f[1]			\ +		   || (X##_f[1] == Y##_f[1]		\ +		       && X##_f[0] > Y##_f[0])))))) + +#define _FP_FRAC_GE_4(X, Y)				\ +  (X##_f[3] > Y##_f[3]					\ +   || (X##_f[3] == Y##_f[3]				\ +       && (X##_f[2] > Y##_f[2]				\ +	   || (X##_f[2] == Y##_f[2]			\ +	       && (X##_f[1] > Y##_f[1]			\ +		   || (X##_f[1] == Y##_f[1]		\ +		       && X##_f[0] >= Y##_f[0])))))) + + +#define _FP_FRAC_CLZ_4(R, X)			\ +  do						\ +    {						\ +      if (X##_f[3])				\ +	__FP_CLZ ((R), X##_f[3]);		\ +      else if (X##_f[2])			\ +	{					\ +	  __FP_CLZ ((R), X##_f[2]);		\ +	  (R) += _FP_W_TYPE_SIZE;		\ +	}					\ +      else if (X##_f[1])			\ +	{					\ +	  __FP_CLZ ((R), X##_f[1]);		\ +	  (R) += _FP_W_TYPE_SIZE*2;		\ +	}					\ +      else					\ +	{					\ +	  __FP_CLZ ((R), X##_f[0]);		\ +	  (R) += _FP_W_TYPE_SIZE*3;		\ +	}					\ +    }						\ +  while (0) + + +#define _FP_UNPACK_RAW_4(fs, X, val)			\ +  do							\ +    {							\ +      union _FP_UNION_##fs _FP_UNPACK_RAW_4_flo;	\ +      _FP_UNPACK_RAW_4_flo.flt = (val);			\ +      X##_f[0] = _FP_UNPACK_RAW_4_flo.bits.frac0;	\ +      X##_f[1] = _FP_UNPACK_RAW_4_flo.bits.frac1;	\ +      X##_f[2] = _FP_UNPACK_RAW_4_flo.bits.frac2;	\ +      X##_f[3] = _FP_UNPACK_RAW_4_flo.bits.frac3;	\ +      X##_e  = _FP_UNPACK_RAW_4_flo.bits.exp;		\ +      X##_s  = _FP_UNPACK_RAW_4_flo.bits.sign;		\ +    }							\ +  while (0) + +#define _FP_UNPACK_RAW_4_P(fs, X, val)			\ +  do							\ +    {							\ +      union _FP_UNION_##fs *_FP_UNPACK_RAW_4_P_flo	\ +	= (union _FP_UNION_##fs *) (val);		\ +							\ +      X##_f[0] = _FP_UNPACK_RAW_4_P_flo->bits.frac0;	\ +      X##_f[1] = _FP_UNPACK_RAW_4_P_flo->bits.frac1;	\ +      X##_f[2] = _FP_UNPACK_RAW_4_P_flo->bits.frac2;	\ +      X##_f[3] = _FP_UNPACK_RAW_4_P_flo->bits.frac3;	\ +      X##_e  = _FP_UNPACK_RAW_4_P_flo->bits.exp;	\ +      X##_s  = _FP_UNPACK_RAW_4_P_flo->bits.sign;	\ +    }							\ +  while (0) + +#define _FP_PACK_RAW_4(fs, val, X)		\ +  do						\ +    {						\ +      union _FP_UNION_##fs _FP_PACK_RAW_4_flo;	\ +      _FP_PACK_RAW_4_flo.bits.frac0 = X##_f[0];	\ +      _FP_PACK_RAW_4_flo.bits.frac1 = X##_f[1];	\ +      _FP_PACK_RAW_4_flo.bits.frac2 = X##_f[2];	\ +      _FP_PACK_RAW_4_flo.bits.frac3 = X##_f[3];	\ +      _FP_PACK_RAW_4_flo.bits.exp   = X##_e;	\ +      _FP_PACK_RAW_4_flo.bits.sign  = X##_s;	\ +      (val) = _FP_PACK_RAW_4_flo.flt;		\ +    }						\ +  while (0) + +#define _FP_PACK_RAW_4_P(fs, val, X)			\ +  do							\ +    {							\ +      union _FP_UNION_##fs *_FP_PACK_RAW_4_P_flo	\ +	= (union _FP_UNION_##fs *) (val);		\ +							\ +      _FP_PACK_RAW_4_P_flo->bits.frac0 = X##_f[0];	\ +      _FP_PACK_RAW_4_P_flo->bits.frac1 = X##_f[1];	\ +      _FP_PACK_RAW_4_P_flo->bits.frac2 = X##_f[2];	\ +      _FP_PACK_RAW_4_P_flo->bits.frac3 = X##_f[3];	\ +      _FP_PACK_RAW_4_P_flo->bits.exp   = X##_e;		\ +      _FP_PACK_RAW_4_P_flo->bits.sign  = X##_s;		\ +    }							\ +  while (0) + +/* Multiplication algorithms: */ + +/* Given a 1W * 1W => 2W primitive, do the extended multiplication.  */ + +#define _FP_MUL_MEAT_DW_4_wide(wfracbits, R, X, Y, doit)		\ +  do									\ +    {									\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_DW_4_wide_b);			\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_DW_4_wide_c);			\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_DW_4_wide_d);			\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_DW_4_wide_e);			\ +      _FP_FRAC_DECL_2 (_FP_MUL_MEAT_DW_4_wide_f);			\ +									\ +      doit (_FP_FRAC_WORD_8 (R, 1), _FP_FRAC_WORD_8 (R, 0),		\ +	    X##_f[0], Y##_f[0]);					\ +      doit (_FP_MUL_MEAT_DW_4_wide_b_f1, _FP_MUL_MEAT_DW_4_wide_b_f0,	\ +	    X##_f[0], Y##_f[1]);					\ +      doit (_FP_MUL_MEAT_DW_4_wide_c_f1, _FP_MUL_MEAT_DW_4_wide_c_f0,	\ +	    X##_f[1], Y##_f[0]);					\ +      doit (_FP_MUL_MEAT_DW_4_wide_d_f1, _FP_MUL_MEAT_DW_4_wide_d_f0,	\ +	    X##_f[1], Y##_f[1]);					\ +      doit (_FP_MUL_MEAT_DW_4_wide_e_f1, _FP_MUL_MEAT_DW_4_wide_e_f0,	\ +	    X##_f[0], Y##_f[2]);					\ +      doit (_FP_MUL_MEAT_DW_4_wide_f_f1, _FP_MUL_MEAT_DW_4_wide_f_f0,	\ +	    X##_f[2], Y##_f[0]);					\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 3), _FP_FRAC_WORD_8 (R, 2),	\ +		       _FP_FRAC_WORD_8 (R, 1), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_b_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_b_f0,			\ +		       0, 0, _FP_FRAC_WORD_8 (R, 1));			\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 3), _FP_FRAC_WORD_8 (R, 2),	\ +		       _FP_FRAC_WORD_8 (R, 1), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_c_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_c_f0,			\ +		       _FP_FRAC_WORD_8 (R, 3), _FP_FRAC_WORD_8 (R, 2),	\ +		       _FP_FRAC_WORD_8 (R, 1));				\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 4), _FP_FRAC_WORD_8 (R, 3),	\ +		       _FP_FRAC_WORD_8 (R, 2), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_d_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_d_f0,			\ +		       0, _FP_FRAC_WORD_8 (R, 3), _FP_FRAC_WORD_8 (R, 2)); \ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 4), _FP_FRAC_WORD_8 (R, 3),	\ +		       _FP_FRAC_WORD_8 (R, 2), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_e_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_e_f0,			\ +		       _FP_FRAC_WORD_8 (R, 4), _FP_FRAC_WORD_8 (R, 3),	\ +		       _FP_FRAC_WORD_8 (R, 2));				\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 4), _FP_FRAC_WORD_8 (R, 3),	\ +		       _FP_FRAC_WORD_8 (R, 2), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_f_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_f_f0,			\ +		       _FP_FRAC_WORD_8 (R, 4), _FP_FRAC_WORD_8 (R, 3),	\ +		       _FP_FRAC_WORD_8 (R, 2));				\ +      doit (_FP_MUL_MEAT_DW_4_wide_b_f1,				\ +	    _FP_MUL_MEAT_DW_4_wide_b_f0, X##_f[0], Y##_f[3]);		\ +      doit (_FP_MUL_MEAT_DW_4_wide_c_f1,				\ +	    _FP_MUL_MEAT_DW_4_wide_c_f0, X##_f[3], Y##_f[0]);		\ +      doit (_FP_MUL_MEAT_DW_4_wide_d_f1, _FP_MUL_MEAT_DW_4_wide_d_f0,	\ +	    X##_f[1], Y##_f[2]);					\ +      doit (_FP_MUL_MEAT_DW_4_wide_e_f1, _FP_MUL_MEAT_DW_4_wide_e_f0,	\ +	    X##_f[2], Y##_f[1]);					\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 5), _FP_FRAC_WORD_8 (R, 4),	\ +		       _FP_FRAC_WORD_8 (R, 3), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_b_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_b_f0,			\ +		       0, _FP_FRAC_WORD_8 (R, 4), _FP_FRAC_WORD_8 (R, 3)); \ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 5), _FP_FRAC_WORD_8 (R, 4),	\ +		       _FP_FRAC_WORD_8 (R, 3), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_c_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_c_f0,			\ +		       _FP_FRAC_WORD_8 (R, 5), _FP_FRAC_WORD_8 (R, 4),	\ +		       _FP_FRAC_WORD_8 (R, 3));				\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 5), _FP_FRAC_WORD_8 (R, 4),	\ +		       _FP_FRAC_WORD_8 (R, 3), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_d_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_d_f0,			\ +		       _FP_FRAC_WORD_8 (R, 5), _FP_FRAC_WORD_8 (R, 4),	\ +		       _FP_FRAC_WORD_8 (R, 3));				\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 5), _FP_FRAC_WORD_8 (R, 4),	\ +		       _FP_FRAC_WORD_8 (R, 3), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_e_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_e_f0,			\ +		       _FP_FRAC_WORD_8 (R, 5), _FP_FRAC_WORD_8 (R, 4),	\ +		       _FP_FRAC_WORD_8 (R, 3));				\ +      doit (_FP_MUL_MEAT_DW_4_wide_b_f1, _FP_MUL_MEAT_DW_4_wide_b_f0,	\ +	    X##_f[2], Y##_f[2]);					\ +      doit (_FP_MUL_MEAT_DW_4_wide_c_f1, _FP_MUL_MEAT_DW_4_wide_c_f0,	\ +	    X##_f[1], Y##_f[3]);					\ +      doit (_FP_MUL_MEAT_DW_4_wide_d_f1, _FP_MUL_MEAT_DW_4_wide_d_f0,	\ +	    X##_f[3], Y##_f[1]);					\ +      doit (_FP_MUL_MEAT_DW_4_wide_e_f1, _FP_MUL_MEAT_DW_4_wide_e_f0,	\ +	    X##_f[2], Y##_f[3]);					\ +      doit (_FP_MUL_MEAT_DW_4_wide_f_f1, _FP_MUL_MEAT_DW_4_wide_f_f0,	\ +	    X##_f[3], Y##_f[2]);					\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 6), _FP_FRAC_WORD_8 (R, 5),	\ +		       _FP_FRAC_WORD_8 (R, 4), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_b_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_b_f0,			\ +		       0, _FP_FRAC_WORD_8 (R, 5), _FP_FRAC_WORD_8 (R, 4)); \ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 6), _FP_FRAC_WORD_8 (R, 5),	\ +		       _FP_FRAC_WORD_8 (R, 4), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_c_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_c_f0,			\ +		       _FP_FRAC_WORD_8 (R, 6), _FP_FRAC_WORD_8 (R, 5),	\ +		       _FP_FRAC_WORD_8 (R, 4));				\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 6), _FP_FRAC_WORD_8 (R, 5),	\ +		       _FP_FRAC_WORD_8 (R, 4), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_d_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_d_f0,			\ +		       _FP_FRAC_WORD_8 (R, 6), _FP_FRAC_WORD_8 (R, 5),	\ +		       _FP_FRAC_WORD_8 (R, 4));				\ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 7), _FP_FRAC_WORD_8 (R, 6),	\ +		       _FP_FRAC_WORD_8 (R, 5), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_e_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_e_f0,			\ +		       0, _FP_FRAC_WORD_8 (R, 6), _FP_FRAC_WORD_8 (R, 5)); \ +      __FP_FRAC_ADD_3 (_FP_FRAC_WORD_8 (R, 7), _FP_FRAC_WORD_8 (R, 6),	\ +		       _FP_FRAC_WORD_8 (R, 5), 0,			\ +		       _FP_MUL_MEAT_DW_4_wide_f_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_f_f0,			\ +		       _FP_FRAC_WORD_8 (R, 7), _FP_FRAC_WORD_8 (R, 6),	\ +		       _FP_FRAC_WORD_8 (R, 5));				\ +      doit (_FP_MUL_MEAT_DW_4_wide_b_f1, _FP_MUL_MEAT_DW_4_wide_b_f0,	\ +	    X##_f[3], Y##_f[3]);					\ +      __FP_FRAC_ADD_2 (_FP_FRAC_WORD_8 (R, 7), _FP_FRAC_WORD_8 (R, 6),	\ +		       _FP_MUL_MEAT_DW_4_wide_b_f1,			\ +		       _FP_MUL_MEAT_DW_4_wide_b_f0,			\ +		       _FP_FRAC_WORD_8 (R, 7), _FP_FRAC_WORD_8 (R, 6));	\ +    }									\ +  while (0) + +#define _FP_MUL_MEAT_4_wide(wfracbits, R, X, Y, doit)			\ +  do									\ +    {									\ +      _FP_FRAC_DECL_8 (_FP_MUL_MEAT_4_wide_z);				\ +									\ +      _FP_MUL_MEAT_DW_4_wide ((wfracbits), _FP_MUL_MEAT_4_wide_z,	\ +			      X, Y, doit);				\ +									\ +      /* Normalize since we know where the msb of the multiplicands	\ +	 were (bit B), we know that the msb of the of the product is	\ +	 at either 2B or 2B-1.  */					\ +      _FP_FRAC_SRS_8 (_FP_MUL_MEAT_4_wide_z, (wfracbits)-1,		\ +		      2*(wfracbits));					\ +      __FP_FRAC_SET_4 (R, _FP_FRAC_WORD_8 (_FP_MUL_MEAT_4_wide_z, 3),	\ +		       _FP_FRAC_WORD_8 (_FP_MUL_MEAT_4_wide_z, 2),	\ +		       _FP_FRAC_WORD_8 (_FP_MUL_MEAT_4_wide_z, 1),	\ +		       _FP_FRAC_WORD_8 (_FP_MUL_MEAT_4_wide_z, 0));	\ +    }									\ +  while (0) + +#define _FP_MUL_MEAT_DW_4_gmp(wfracbits, R, X, Y)	\ +  do							\ +    {							\ +      mpn_mul_n (R##_f, _x_f, _y_f, 4);			\ +    }							\ +  while (0) + +#define _FP_MUL_MEAT_4_gmp(wfracbits, R, X, Y)				\ +  do									\ +    {									\ +      _FP_FRAC_DECL_8 (_FP_MUL_MEAT_4_gmp_z);				\ +									\ +      _FP_MUL_MEAT_DW_4_gmp ((wfracbits), _FP_MUL_MEAT_4_gmp_z, X, Y);	\ +									\ +      /* Normalize since we know where the msb of the multiplicands	\ +	 were (bit B), we know that the msb of the of the product is	\ +	 at either 2B or 2B-1.  */					\ +      _FP_FRAC_SRS_8 (_FP_MUL_MEAT_4_gmp_z, (wfracbits)-1,		\ +		      2*(wfracbits));					\ +      __FP_FRAC_SET_4 (R, _FP_FRAC_WORD_8 (_FP_MUL_MEAT_4_gmp_z, 3),	\ +		       _FP_FRAC_WORD_8 (_FP_MUL_MEAT_4_gmp_z, 2),	\ +		       _FP_FRAC_WORD_8 (_FP_MUL_MEAT_4_gmp_z, 1),	\ +		       _FP_FRAC_WORD_8 (_FP_MUL_MEAT_4_gmp_z, 0));	\ +    }									\ +  while (0) + +/* Helper utility for _FP_DIV_MEAT_4_udiv: + * pppp = m * nnn.  */ +#define umul_ppppmnnn(p3, p2, p1, p0, m, n2, n1, n0)	\ +  do							\ +    {							\ +      UWtype umul_ppppmnnn_t;				\ +      umul_ppmm (p1, p0, m, n0);			\ +      umul_ppmm (p2, umul_ppppmnnn_t, m, n1);		\ +      __FP_FRAC_ADDI_2 (p2, p1, umul_ppppmnnn_t);	\ +      umul_ppmm (p3, umul_ppppmnnn_t, m, n2);		\ +      __FP_FRAC_ADDI_2 (p3, p2, umul_ppppmnnn_t);	\ +    }							\ +  while (0) + +/* Division algorithms: */ + +#define _FP_DIV_MEAT_4_udiv(fs, R, X, Y)				\ +  do									\ +    {									\ +      int _FP_DIV_MEAT_4_udiv_i;					\ +      _FP_FRAC_DECL_4 (_FP_DIV_MEAT_4_udiv_n);				\ +      _FP_FRAC_DECL_4 (_FP_DIV_MEAT_4_udiv_m);				\ +      _FP_FRAC_SET_4 (_FP_DIV_MEAT_4_udiv_n, _FP_ZEROFRAC_4);		\ +      if (_FP_FRAC_GE_4 (X, Y))						\ +	{								\ +	  _FP_DIV_MEAT_4_udiv_n_f[3]					\ +	    = X##_f[0] << (_FP_W_TYPE_SIZE - 1);			\ +	  _FP_FRAC_SRL_4 (X, 1);					\ +	}								\ +      else								\ +	R##_e--;							\ +									\ +      /* Normalize, i.e. make the most significant bit of the		\ +	 denominator set.  */						\ +      _FP_FRAC_SLL_4 (Y, _FP_WFRACXBITS_##fs);				\ +									\ +      for (_FP_DIV_MEAT_4_udiv_i = 3; ; _FP_DIV_MEAT_4_udiv_i--)	\ +	{								\ +	  if (X##_f[3] == Y##_f[3])					\ +	    {								\ +	      /* This is a special case, not an optimization		\ +		 (X##_f[3]/Y##_f[3] would not fit into UWtype).		\ +		 As X## is guaranteed to be < Y,			\ +		 R##_f[_FP_DIV_MEAT_4_udiv_i] can be either		\ +		 (UWtype)-1 or (UWtype)-2.  */				\ +	      R##_f[_FP_DIV_MEAT_4_udiv_i] = -1;			\ +	      if (!_FP_DIV_MEAT_4_udiv_i)				\ +		break;							\ +	      __FP_FRAC_SUB_4 (X##_f[3], X##_f[2], X##_f[1], X##_f[0],	\ +			       Y##_f[2], Y##_f[1], Y##_f[0], 0,		\ +			       X##_f[2], X##_f[1], X##_f[0],		\ +			       _FP_DIV_MEAT_4_udiv_n_f[_FP_DIV_MEAT_4_udiv_i]); \ +	      _FP_FRAC_SUB_4 (X, Y, X);					\ +	      if (X##_f[3] > Y##_f[3])					\ +		{							\ +		  R##_f[_FP_DIV_MEAT_4_udiv_i] = -2;			\ +		  _FP_FRAC_ADD_4 (X, Y, X);				\ +		}							\ +	    }								\ +	  else								\ +	    {								\ +	      udiv_qrnnd (R##_f[_FP_DIV_MEAT_4_udiv_i],			\ +			  X##_f[3], X##_f[3], X##_f[2], Y##_f[3]);	\ +	      umul_ppppmnnn (_FP_DIV_MEAT_4_udiv_m_f[3],		\ +			     _FP_DIV_MEAT_4_udiv_m_f[2],		\ +			     _FP_DIV_MEAT_4_udiv_m_f[1],		\ +			     _FP_DIV_MEAT_4_udiv_m_f[0],		\ +			     R##_f[_FP_DIV_MEAT_4_udiv_i],		\ +			     Y##_f[2], Y##_f[1], Y##_f[0]);		\ +	      X##_f[2] = X##_f[1];					\ +	      X##_f[1] = X##_f[0];					\ +	      X##_f[0]							\ +		= _FP_DIV_MEAT_4_udiv_n_f[_FP_DIV_MEAT_4_udiv_i];	\ +	      if (_FP_FRAC_GT_4 (_FP_DIV_MEAT_4_udiv_m, X))		\ +		{							\ +		  R##_f[_FP_DIV_MEAT_4_udiv_i]--;			\ +		  _FP_FRAC_ADD_4 (X, Y, X);				\ +		  if (_FP_FRAC_GE_4 (X, Y)				\ +		      && _FP_FRAC_GT_4 (_FP_DIV_MEAT_4_udiv_m, X))	\ +		    {							\ +		      R##_f[_FP_DIV_MEAT_4_udiv_i]--;			\ +		      _FP_FRAC_ADD_4 (X, Y, X);				\ +		    }							\ +		}							\ +	      _FP_FRAC_DEC_4 (X, _FP_DIV_MEAT_4_udiv_m);		\ +	      if (!_FP_DIV_MEAT_4_udiv_i)				\ +		{							\ +		  if (!_FP_FRAC_EQ_4 (X, _FP_DIV_MEAT_4_udiv_m))	\ +		    R##_f[0] |= _FP_WORK_STICKY;			\ +		  break;						\ +		}							\ +	    }								\ +	}								\ +    }									\ +  while (0) + + +/* Square root algorithms: +   We have just one right now, maybe Newton approximation +   should be added for those machines where division is fast.  */ + +#define _FP_SQRT_MEAT_4(R, S, T, X, q)					\ +  do									\ +    {									\ +      while (q)								\ +	{								\ +	  T##_f[3] = S##_f[3] + (q);					\ +	  if (T##_f[3] <= X##_f[3])					\ +	    {								\ +	      S##_f[3] = T##_f[3] + (q);				\ +	      X##_f[3] -= T##_f[3];					\ +	      R##_f[3] += (q);						\ +	    }								\ +	  _FP_FRAC_SLL_4 (X, 1);					\ +	  (q) >>= 1;							\ +	}								\ +      (q) = (_FP_W_TYPE) 1 << (_FP_W_TYPE_SIZE - 1);			\ +      while (q)								\ +	{								\ +	  T##_f[2] = S##_f[2] + (q);					\ +	  T##_f[3] = S##_f[3];						\ +	  if (T##_f[3] < X##_f[3]					\ +	      || (T##_f[3] == X##_f[3] && T##_f[2] <= X##_f[2]))	\ +	    {								\ +	      S##_f[2] = T##_f[2] + (q);				\ +	      S##_f[3] += (T##_f[2] > S##_f[2]);			\ +	      __FP_FRAC_DEC_2 (X##_f[3], X##_f[2],			\ +			       T##_f[3], T##_f[2]);			\ +	      R##_f[2] += (q);						\ +	    }								\ +	  _FP_FRAC_SLL_4 (X, 1);					\ +	  (q) >>= 1;							\ +	}								\ +      (q) = (_FP_W_TYPE) 1 << (_FP_W_TYPE_SIZE - 1);			\ +      while (q)								\ +	{								\ +	  T##_f[1] = S##_f[1] + (q);					\ +	  T##_f[2] = S##_f[2];						\ +	  T##_f[3] = S##_f[3];						\ +	  if (T##_f[3] < X##_f[3]					\ +	      || (T##_f[3] == X##_f[3]					\ +		  && (T##_f[2] < X##_f[2]				\ +		      || (T##_f[2] == X##_f[2]				\ +			  && T##_f[1] <= X##_f[1]))))			\ +	    {								\ +	      S##_f[1] = T##_f[1] + (q);				\ +	      S##_f[2] += (T##_f[1] > S##_f[1]);			\ +	      S##_f[3] += (T##_f[2] > S##_f[2]);			\ +	      __FP_FRAC_DEC_3 (X##_f[3], X##_f[2], X##_f[1],		\ +			       T##_f[3], T##_f[2], T##_f[1]);		\ +	      R##_f[1] += (q);						\ +	    }								\ +	  _FP_FRAC_SLL_4 (X, 1);					\ +	  (q) >>= 1;							\ +	}								\ +      (q) = (_FP_W_TYPE) 1 << (_FP_W_TYPE_SIZE - 1);			\ +      while ((q) != _FP_WORK_ROUND)					\ +	{								\ +	  T##_f[0] = S##_f[0] + (q);					\ +	  T##_f[1] = S##_f[1];						\ +	  T##_f[2] = S##_f[2];						\ +	  T##_f[3] = S##_f[3];						\ +	  if (_FP_FRAC_GE_4 (X, T))					\ +	    {								\ +	      S##_f[0] = T##_f[0] + (q);				\ +	      S##_f[1] += (T##_f[0] > S##_f[0]);			\ +	      S##_f[2] += (T##_f[1] > S##_f[1]);			\ +	      S##_f[3] += (T##_f[2] > S##_f[2]);			\ +	      _FP_FRAC_DEC_4 (X, T);					\ +	      R##_f[0] += (q);						\ +	    }								\ +	  _FP_FRAC_SLL_4 (X, 1);					\ +	  (q) >>= 1;							\ +	}								\ +      if (!_FP_FRAC_ZEROP_4 (X))					\ +	{								\ +	  if (_FP_FRAC_GT_4 (X, S))					\ +	    R##_f[0] |= _FP_WORK_ROUND;					\ +	  R##_f[0] |= _FP_WORK_STICKY;					\ +	}								\ +    }									\ +  while (0) + + +/* Internals.  */ + +#define __FP_FRAC_SET_4(X, I3, I2, I1, I0)			\ +  (X##_f[3] = I3, X##_f[2] = I2, X##_f[1] = I1, X##_f[0] = I0) + +#ifndef __FP_FRAC_ADD_3 +# define __FP_FRAC_ADD_3(r2, r1, r0, x2, x1, x0, y2, y1, y0)	\ +  do								\ +    {								\ +      _FP_W_TYPE __FP_FRAC_ADD_3_c1, __FP_FRAC_ADD_3_c2;	\ +      r0 = x0 + y0;						\ +      __FP_FRAC_ADD_3_c1 = r0 < x0;				\ +      r1 = x1 + y1;						\ +      __FP_FRAC_ADD_3_c2 = r1 < x1;				\ +      r1 += __FP_FRAC_ADD_3_c1;					\ +      __FP_FRAC_ADD_3_c2 |= r1 < __FP_FRAC_ADD_3_c1;		\ +      r2 = x2 + y2 + __FP_FRAC_ADD_3_c2;			\ +    }								\ +  while (0) +#endif + +#ifndef __FP_FRAC_ADD_4 +# define __FP_FRAC_ADD_4(r3, r2, r1, r0, x3, x2, x1, x0, y3, y2, y1, y0) \ +  do									\ +    {									\ +      _FP_W_TYPE __FP_FRAC_ADD_4_c1, __FP_FRAC_ADD_4_c2;		\ +      _FP_W_TYPE __FP_FRAC_ADD_4_c3;					\ +      r0 = x0 + y0;							\ +      __FP_FRAC_ADD_4_c1 = r0 < x0;					\ +      r1 = x1 + y1;							\ +      __FP_FRAC_ADD_4_c2 = r1 < x1;					\ +      r1 += __FP_FRAC_ADD_4_c1;						\ +      __FP_FRAC_ADD_4_c2 |= r1 < __FP_FRAC_ADD_4_c1;			\ +      r2 = x2 + y2;							\ +      __FP_FRAC_ADD_4_c3 = r2 < x2;					\ +      r2 += __FP_FRAC_ADD_4_c2;						\ +      __FP_FRAC_ADD_4_c3 |= r2 < __FP_FRAC_ADD_4_c2;			\ +      r3 = x3 + y3 + __FP_FRAC_ADD_4_c3;				\ +    }									\ +  while (0) +#endif + +#ifndef __FP_FRAC_SUB_3 +# define __FP_FRAC_SUB_3(r2, r1, r0, x2, x1, x0, y2, y1, y0)	\ +  do								\ +    {								\ +      _FP_W_TYPE __FP_FRAC_SUB_3_c1, __FP_FRAC_SUB_3_c2;	\ +      r0 = x0 - y0;						\ +      __FP_FRAC_SUB_3_c1 = r0 > x0;				\ +      r1 = x1 - y1;						\ +      __FP_FRAC_SUB_3_c2 = r1 > x1;				\ +      r1 -= __FP_FRAC_SUB_3_c1;					\ +      __FP_FRAC_SUB_3_c2 |= __FP_FRAC_SUB_3_c1 && (y1 == x1);	\ +      r2 = x2 - y2 - __FP_FRAC_SUB_3_c2;			\ +    }								\ +  while (0) +#endif + +#ifndef __FP_FRAC_SUB_4 +# define __FP_FRAC_SUB_4(r3, r2, r1, r0, x3, x2, x1, x0, y3, y2, y1, y0) \ +  do									\ +    {									\ +      _FP_W_TYPE __FP_FRAC_SUB_4_c1, __FP_FRAC_SUB_4_c2;		\ +      _FP_W_TYPE __FP_FRAC_SUB_4_c3;					\ +      r0 = x0 - y0;							\ +      __FP_FRAC_SUB_4_c1 = r0 > x0;					\ +      r1 = x1 - y1;							\ +      __FP_FRAC_SUB_4_c2 = r1 > x1;					\ +      r1 -= __FP_FRAC_SUB_4_c1;						\ +      __FP_FRAC_SUB_4_c2 |= __FP_FRAC_SUB_4_c1 && (y1 == x1);		\ +      r2 = x2 - y2;							\ +      __FP_FRAC_SUB_4_c3 = r2 > x2;					\ +      r2 -= __FP_FRAC_SUB_4_c2;						\ +      __FP_FRAC_SUB_4_c3 |= __FP_FRAC_SUB_4_c2 && (y2 == x2);		\ +      r3 = x3 - y3 - __FP_FRAC_SUB_4_c3;				\ +    }									\ +  while (0) +#endif + +#ifndef __FP_FRAC_DEC_3 +# define __FP_FRAC_DEC_3(x2, x1, x0, y2, y1, y0)		\ +  do								\ +    {								\ +      UWtype __FP_FRAC_DEC_3_t0, __FP_FRAC_DEC_3_t1;		\ +      UWtype __FP_FRAC_DEC_3_t2;				\ +      __FP_FRAC_DEC_3_t0 = x0;					\ +      __FP_FRAC_DEC_3_t1 = x1;					\ +      __FP_FRAC_DEC_3_t2 = x2;					\ +      __FP_FRAC_SUB_3 (x2, x1, x0, __FP_FRAC_DEC_3_t2,		\ +		       __FP_FRAC_DEC_3_t1, __FP_FRAC_DEC_3_t0,	\ +		       y2, y1, y0);				\ +    }								\ +  while (0) +#endif + +#ifndef __FP_FRAC_DEC_4 +# define __FP_FRAC_DEC_4(x3, x2, x1, x0, y3, y2, y1, y0)	\ +  do								\ +    {								\ +      UWtype __FP_FRAC_DEC_4_t0, __FP_FRAC_DEC_4_t1;		\ +      UWtype __FP_FRAC_DEC_4_t2, __FP_FRAC_DEC_4_t3;		\ +      __FP_FRAC_DEC_4_t0 = x0;					\ +      __FP_FRAC_DEC_4_t1 = x1;					\ +      __FP_FRAC_DEC_4_t2 = x2;					\ +      __FP_FRAC_DEC_4_t3 = x3;					\ +      __FP_FRAC_SUB_4 (x3, x2, x1, x0, __FP_FRAC_DEC_4_t3,	\ +		       __FP_FRAC_DEC_4_t2, __FP_FRAC_DEC_4_t1,	\ +		       __FP_FRAC_DEC_4_t0, y3, y2, y1, y0);	\ +    }								\ +  while (0) +#endif + +#ifndef __FP_FRAC_ADDI_4 +# define __FP_FRAC_ADDI_4(x3, x2, x1, x0, i)		\ +  do							\ +    {							\ +      UWtype __FP_FRAC_ADDI_4_t;			\ +      __FP_FRAC_ADDI_4_t = ((x0 += i) < i);		\ +      x1 += __FP_FRAC_ADDI_4_t;				\ +      __FP_FRAC_ADDI_4_t = (x1 < __FP_FRAC_ADDI_4_t);	\ +      x2 += __FP_FRAC_ADDI_4_t;				\ +      __FP_FRAC_ADDI_4_t = (x2 < __FP_FRAC_ADDI_4_t);	\ +      x3 += __FP_FRAC_ADDI_4_t;				\ +    }							\ +  while (0) +#endif + +/* Convert FP values between word sizes. This appears to be more +   complicated than I'd have expected it to be, so these might be +   wrong... These macros are in any case somewhat bogus because they +   use information about what various FRAC_n variables look like +   internally [eg, that 2 word vars are X_f0 and x_f1]. But so do +   the ones in op-2.h and op-1.h.  */ +#define _FP_FRAC_COPY_1_4(D, S)		(D##_f = S##_f[0]) + +#define _FP_FRAC_COPY_2_4(D, S)			\ +  do						\ +    {						\ +      D##_f0 = S##_f[0];			\ +      D##_f1 = S##_f[1];			\ +    }						\ +  while (0) + +/* Assembly/disassembly for converting to/from integral types. +   No shifting or overflow handled here.  */ +/* Put the FP value X into r, which is an integer of size rsize.  */ +#define _FP_FRAC_ASSEMBLE_4(r, X, rsize)				\ +  do									\ +    {									\ +      if ((rsize) <= _FP_W_TYPE_SIZE)					\ +	(r) = X##_f[0];							\ +	else if ((rsize) <= 2*_FP_W_TYPE_SIZE)				\ +	{								\ +	  (r) = X##_f[1];						\ +	  (r) = ((rsize) <= _FP_W_TYPE_SIZE				\ +		 ? 0							\ +		 : (r) << _FP_W_TYPE_SIZE);				\ +	  (r) += X##_f[0];						\ +	}								\ +      else								\ +	{								\ +	  /* I'm feeling lazy so we deal with int == 3words		\ +	     (implausible) and int == 4words as a single case.  */	\ +	  (r) = X##_f[3];						\ +	  (r) = ((rsize) <= _FP_W_TYPE_SIZE				\ +		 ? 0							\ +		 : (r) << _FP_W_TYPE_SIZE);				\ +	  (r) += X##_f[2];						\ +	  (r) = ((rsize) <= _FP_W_TYPE_SIZE				\ +		 ? 0							\ +		 : (r) << _FP_W_TYPE_SIZE);				\ +	  (r) += X##_f[1];						\ +	  (r) = ((rsize) <= _FP_W_TYPE_SIZE				\ +		 ? 0							\ +		 : (r) << _FP_W_TYPE_SIZE);				\ +	  (r) += X##_f[0];						\ +	}								\ +    }									\ +  while (0) + +/* "No disassemble Number Five!" */ +/* Move an integer of size rsize into X's fractional part. We rely on +   the _f[] array consisting of words of size _FP_W_TYPE_SIZE to avoid +   having to mask the values we store into it.  */ +#define _FP_FRAC_DISASSEMBLE_4(X, r, rsize)	\ +  do						\ +    {						\ +      X##_f[0] = (r);				\ +      X##_f[1] = ((rsize) <= _FP_W_TYPE_SIZE	\ +		  ? 0				\ +		  : (r) >> _FP_W_TYPE_SIZE);	\ +      X##_f[2] = ((rsize) <= 2*_FP_W_TYPE_SIZE	\ +		  ? 0				\ +		  : (r) >> 2*_FP_W_TYPE_SIZE);	\ +      X##_f[3] = ((rsize) <= 3*_FP_W_TYPE_SIZE	\ +		  ? 0				\ +		  : (r) >> 3*_FP_W_TYPE_SIZE);	\ +    }						\ +  while (0) + +#define _FP_FRAC_COPY_4_1(D, S)			\ +  do						\ +    {						\ +      D##_f[0] = S##_f;				\ +      D##_f[1] = D##_f[2] = D##_f[3] = 0;	\ +    }						\ +  while (0) + +#define _FP_FRAC_COPY_4_2(D, S)			\ +  do						\ +    {						\ +      D##_f[0] = S##_f0;			\ +      D##_f[1] = S##_f1;			\ +      D##_f[2] = D##_f[3] = 0;			\ +    }						\ +  while (0) + +#define _FP_FRAC_COPY_4_4(D, S)	_FP_FRAC_COPY_4 (D, S) + +#endif /* !SOFT_FP_OP_4_H */ diff --git a/libc/sysdeps/linux/sparc64/soft-fp/op-8.h b/libc/sysdeps/linux/sparc64/soft-fp/op-8.h new file mode 100644 index 000000000..1f424e102 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/op-8.h @@ -0,0 +1,150 @@ +/* Software floating-point emulation. +   Basic eight-word fraction declaration and manipulation. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com), +		  Jakub Jelinek (jj@ultra.linux.cz) and +		  Peter Maydell (pmaydell@chiark.greenend.org.uk). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 of the License, or (at your option) any later version. + +   In addition to the permissions in the GNU Lesser General Public +   License, the Free Software Foundation gives you unlimited +   permission to link the compiled version of this file into +   combinations with other programs, and to distribute those +   combinations without any restriction coming from the use of this +   file.  (The Lesser General Public License restrictions do apply in +   other respects; for example, they cover modification of the file, +   and distribution when not linked into a combine executable.) + +   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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef SOFT_FP_OP_8_H +#define SOFT_FP_OP_8_H	1 + +/* We need just a few things from here for op-4, if we ever need some +   other macros, they can be added.  */ +#define _FP_FRAC_DECL_8(X)	_FP_W_TYPE X##_f[8] +#define _FP_FRAC_HIGH_8(X)	(X##_f[7]) +#define _FP_FRAC_LOW_8(X)	(X##_f[0]) +#define _FP_FRAC_WORD_8(X, w)	(X##_f[w]) + +#define _FP_FRAC_SLL_8(X, N)						\ +  do									\ +    {									\ +      _FP_I_TYPE _FP_FRAC_SLL_8_up, _FP_FRAC_SLL_8_down;		\ +      _FP_I_TYPE _FP_FRAC_SLL_8_skip, _FP_FRAC_SLL_8_i;			\ +      _FP_FRAC_SLL_8_skip = (N) / _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SLL_8_up = (N) % _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SLL_8_down = _FP_W_TYPE_SIZE - _FP_FRAC_SLL_8_up;	\ +      if (!_FP_FRAC_SLL_8_up)						\ +	for (_FP_FRAC_SLL_8_i = 7;					\ +	     _FP_FRAC_SLL_8_i >= _FP_FRAC_SLL_8_skip;			\ +	     --_FP_FRAC_SLL_8_i)					\ +	  X##_f[_FP_FRAC_SLL_8_i]					\ +	    = X##_f[_FP_FRAC_SLL_8_i-_FP_FRAC_SLL_8_skip];		\ +      else								\ +	{								\ +	  for (_FP_FRAC_SLL_8_i = 7;					\ +	       _FP_FRAC_SLL_8_i > _FP_FRAC_SLL_8_skip;			\ +	       --_FP_FRAC_SLL_8_i)					\ +	    X##_f[_FP_FRAC_SLL_8_i]					\ +	      = ((X##_f[_FP_FRAC_SLL_8_i-_FP_FRAC_SLL_8_skip]		\ +		  << _FP_FRAC_SLL_8_up)					\ +		 | (X##_f[_FP_FRAC_SLL_8_i-_FP_FRAC_SLL_8_skip-1]	\ +		    >> _FP_FRAC_SLL_8_down));				\ +	  X##_f[_FP_FRAC_SLL_8_i--] = X##_f[0] << _FP_FRAC_SLL_8_up;	\ +	}								\ +      for (; _FP_FRAC_SLL_8_i >= 0; --_FP_FRAC_SLL_8_i)			\ +	X##_f[_FP_FRAC_SLL_8_i] = 0;					\ +    }									\ +  while (0) + +#define _FP_FRAC_SRL_8(X, N)						\ +  do									\ +    {									\ +      _FP_I_TYPE _FP_FRAC_SRL_8_up, _FP_FRAC_SRL_8_down;		\ +      _FP_I_TYPE _FP_FRAC_SRL_8_skip, _FP_FRAC_SRL_8_i;			\ +      _FP_FRAC_SRL_8_skip = (N) / _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SRL_8_down = (N) % _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SRL_8_up = _FP_W_TYPE_SIZE - _FP_FRAC_SRL_8_down;	\ +      if (!_FP_FRAC_SRL_8_down)						\ +	for (_FP_FRAC_SRL_8_i = 0;					\ +	     _FP_FRAC_SRL_8_i <= 7-_FP_FRAC_SRL_8_skip;			\ +	     ++_FP_FRAC_SRL_8_i)					\ +	  X##_f[_FP_FRAC_SRL_8_i]					\ +	    = X##_f[_FP_FRAC_SRL_8_i+_FP_FRAC_SRL_8_skip];		\ +      else								\ +	{								\ +	  for (_FP_FRAC_SRL_8_i = 0;					\ +	       _FP_FRAC_SRL_8_i < 7-_FP_FRAC_SRL_8_skip;		\ +	       ++_FP_FRAC_SRL_8_i)					\ +	    X##_f[_FP_FRAC_SRL_8_i]					\ +	      = ((X##_f[_FP_FRAC_SRL_8_i+_FP_FRAC_SRL_8_skip]		\ +		  >> _FP_FRAC_SRL_8_down)				\ +		 | (X##_f[_FP_FRAC_SRL_8_i+_FP_FRAC_SRL_8_skip+1]	\ +		    << _FP_FRAC_SRL_8_up));				\ +	  X##_f[_FP_FRAC_SRL_8_i++] = X##_f[7] >> _FP_FRAC_SRL_8_down;	\ +	}								\ +      for (; _FP_FRAC_SRL_8_i < 8; ++_FP_FRAC_SRL_8_i)			\ +	X##_f[_FP_FRAC_SRL_8_i] = 0;					\ +    }									\ +  while (0) + + +/* Right shift with sticky-lsb. +   What this actually means is that we do a standard right-shift, +   but that if any of the bits that fall off the right hand side +   were one then we always set the LSbit.  */ +#define _FP_FRAC_SRS_8(X, N, size)					\ +  do									\ +    {									\ +      _FP_I_TYPE _FP_FRAC_SRS_8_up, _FP_FRAC_SRS_8_down;		\ +      _FP_I_TYPE _FP_FRAC_SRS_8_skip, _FP_FRAC_SRS_8_i;			\ +      _FP_W_TYPE _FP_FRAC_SRS_8_s;					\ +      _FP_FRAC_SRS_8_skip = (N) / _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SRS_8_down = (N) % _FP_W_TYPE_SIZE;			\ +      _FP_FRAC_SRS_8_up = _FP_W_TYPE_SIZE - _FP_FRAC_SRS_8_down;	\ +      for (_FP_FRAC_SRS_8_s = _FP_FRAC_SRS_8_i = 0;			\ +	   _FP_FRAC_SRS_8_i < _FP_FRAC_SRS_8_skip;			\ +	   ++_FP_FRAC_SRS_8_i)						\ +	_FP_FRAC_SRS_8_s |= X##_f[_FP_FRAC_SRS_8_i];			\ +      if (!_FP_FRAC_SRS_8_down)						\ +	for (_FP_FRAC_SRS_8_i = 0;					\ +	     _FP_FRAC_SRS_8_i <= 7-_FP_FRAC_SRS_8_skip;			\ +	     ++_FP_FRAC_SRS_8_i)					\ +	  X##_f[_FP_FRAC_SRS_8_i]					\ +	    = X##_f[_FP_FRAC_SRS_8_i+_FP_FRAC_SRS_8_skip];		\ +      else								\ +	{								\ +	  _FP_FRAC_SRS_8_s						\ +	    |= X##_f[_FP_FRAC_SRS_8_i] << _FP_FRAC_SRS_8_up;		\ +	  for (_FP_FRAC_SRS_8_i = 0;					\ +	       _FP_FRAC_SRS_8_i < 7-_FP_FRAC_SRS_8_skip;		\ +	       ++_FP_FRAC_SRS_8_i)					\ +	    X##_f[_FP_FRAC_SRS_8_i]					\ +	      = ((X##_f[_FP_FRAC_SRS_8_i+_FP_FRAC_SRS_8_skip]		\ +		  >> _FP_FRAC_SRS_8_down)				\ +		 | (X##_f[_FP_FRAC_SRS_8_i+_FP_FRAC_SRS_8_skip+1]	\ +		    << _FP_FRAC_SRS_8_up));				\ +	  X##_f[_FP_FRAC_SRS_8_i++] = X##_f[7] >> _FP_FRAC_SRS_8_down;	\ +	}								\ +      for (; _FP_FRAC_SRS_8_i < 8; ++_FP_FRAC_SRS_8_i)			\ +	X##_f[_FP_FRAC_SRS_8_i] = 0;					\ +      /* Don't fix the LSB until the very end when we're sure f[0] is	\ +	 stable.  */							\ +      X##_f[0] |= (_FP_FRAC_SRS_8_s != 0);				\ +    }									\ +  while (0) + +#endif /* !SOFT_FP_OP_8_H */ diff --git a/libc/sysdeps/linux/sparc64/soft-fp/op-common.h b/libc/sysdeps/linux/sparc64/soft-fp/op-common.h new file mode 100644 index 000000000..82b1d09fc --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/op-common.h @@ -0,0 +1,2134 @@ +/* Software floating-point emulation. Common operations. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com), +		  Jakub Jelinek (jj@ultra.linux.cz), +		  David S. Miller (davem@redhat.com) and +		  Peter Maydell (pmaydell@chiark.greenend.org.uk). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 of the License, or (at your option) any later version. + +   In addition to the permissions in the GNU Lesser General Public +   License, the Free Software Foundation gives you unlimited +   permission to link the compiled version of this file into +   combinations with other programs, and to distribute those +   combinations without any restriction coming from the use of this +   file.  (The Lesser General Public License restrictions do apply in +   other respects; for example, they cover modification of the file, +   and distribution when not linked into a combine executable.) + +   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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef SOFT_FP_OP_COMMON_H +#define SOFT_FP_OP_COMMON_H	1 + +#define _FP_DECL(wc, X)						\ +  _FP_I_TYPE X##_c __attribute__ ((unused)) _FP_ZERO_INIT;	\ +  _FP_I_TYPE X##_s __attribute__ ((unused)) _FP_ZERO_INIT;	\ +  _FP_I_TYPE X##_e __attribute__ ((unused)) _FP_ZERO_INIT;	\ +  _FP_FRAC_DECL_##wc (X) + +/* Test whether the qNaN bit denotes a signaling NaN.  */ +#define _FP_FRAC_SNANP(fs, X)				\ +  ((_FP_QNANNEGATEDP)					\ +   ? (_FP_FRAC_HIGH_RAW_##fs (X) & _FP_QNANBIT_##fs)	\ +   : !(_FP_FRAC_HIGH_RAW_##fs (X) & _FP_QNANBIT_##fs)) +#define _FP_FRAC_SNANP_SEMIRAW(fs, X)			\ +  ((_FP_QNANNEGATEDP)					\ +   ? (_FP_FRAC_HIGH_##fs (X) & _FP_QNANBIT_SH_##fs)	\ +   : !(_FP_FRAC_HIGH_##fs (X) & _FP_QNANBIT_SH_##fs)) + +/* Finish truly unpacking a native fp value by classifying the kind +   of fp value and normalizing both the exponent and the fraction.  */ + +#define _FP_UNPACK_CANONICAL(fs, wc, X)				\ +  do								\ +    {								\ +      switch (X##_e)						\ +	{							\ +	default:						\ +	  _FP_FRAC_HIGH_RAW_##fs (X) |= _FP_IMPLBIT_##fs;	\ +	  _FP_FRAC_SLL_##wc (X, _FP_WORKBITS);			\ +	  X##_e -= _FP_EXPBIAS_##fs;				\ +	  X##_c = FP_CLS_NORMAL;				\ +	  break;						\ +								\ +	case 0:							\ +	  if (_FP_FRAC_ZEROP_##wc (X))				\ +	    X##_c = FP_CLS_ZERO;				\ +	  else if (FP_DENORM_ZERO)				\ +	    {							\ +	      X##_c = FP_CLS_ZERO;				\ +	      _FP_FRAC_SET_##wc (X, _FP_ZEROFRAC_##wc);		\ +	      FP_SET_EXCEPTION (FP_EX_DENORM);			\ +	    }							\ +	  else							\ +	    {							\ +	      /* A denormalized number.  */			\ +	      _FP_I_TYPE _FP_UNPACK_CANONICAL_shift;		\ +	      _FP_FRAC_CLZ_##wc (_FP_UNPACK_CANONICAL_shift,	\ +				 X);				\ +	      _FP_UNPACK_CANONICAL_shift -= _FP_FRACXBITS_##fs;	\ +	      _FP_FRAC_SLL_##wc (X, (_FP_UNPACK_CANONICAL_shift \ +				     + _FP_WORKBITS));		\ +	      X##_e -= (_FP_EXPBIAS_##fs - 1			\ +			+ _FP_UNPACK_CANONICAL_shift);		\ +	      X##_c = FP_CLS_NORMAL;				\ +	      FP_SET_EXCEPTION (FP_EX_DENORM);			\ +	    }							\ +	  break;						\ +								\ +	case _FP_EXPMAX_##fs:					\ +	  if (_FP_FRAC_ZEROP_##wc (X))				\ +	    X##_c = FP_CLS_INF;					\ +	  else							\ +	    {							\ +	      X##_c = FP_CLS_NAN;				\ +	      /* Check for signaling NaN.  */			\ +	      if (_FP_FRAC_SNANP (fs, X))			\ +		FP_SET_EXCEPTION (FP_EX_INVALID			\ +				  | FP_EX_INVALID_SNAN);	\ +	    }							\ +	  break;						\ +	}							\ +    }								\ +  while (0) + +/* Finish unpacking an fp value in semi-raw mode: the mantissa is +   shifted by _FP_WORKBITS but the implicit MSB is not inserted and +   other classification is not done.  */ +#define _FP_UNPACK_SEMIRAW(fs, wc, X)	_FP_FRAC_SLL_##wc (X, _FP_WORKBITS) + +/* Check whether a raw or semi-raw input value should be flushed to +   zero, and flush it to zero if so.  */ +#define _FP_CHECK_FLUSH_ZERO(fs, wc, X)			\ +  do							\ +    {							\ +      if (FP_DENORM_ZERO				\ +	  && X##_e == 0					\ +	  && !_FP_FRAC_ZEROP_##wc (X))			\ +	{						\ +	  _FP_FRAC_SET_##wc (X, _FP_ZEROFRAC_##wc);	\ +	  FP_SET_EXCEPTION (FP_EX_DENORM);		\ +	}						\ +    }							\ +  while (0) + +/* A semi-raw value has overflowed to infinity.  Adjust the mantissa +   and exponent appropriately.  */ +#define _FP_OVERFLOW_SEMIRAW(fs, wc, X)			\ +  do							\ +    {							\ +      if (FP_ROUNDMODE == FP_RND_NEAREST		\ +	  || (FP_ROUNDMODE == FP_RND_PINF && !X##_s)	\ +	  || (FP_ROUNDMODE == FP_RND_MINF && X##_s))	\ +	{						\ +	  X##_e = _FP_EXPMAX_##fs;			\ +	  _FP_FRAC_SET_##wc (X, _FP_ZEROFRAC_##wc);	\ +	}						\ +      else						\ +	{						\ +	  X##_e = _FP_EXPMAX_##fs - 1;			\ +	  _FP_FRAC_SET_##wc (X, _FP_MAXFRAC_##wc);	\ +	}						\ +      FP_SET_EXCEPTION (FP_EX_INEXACT);			\ +      FP_SET_EXCEPTION (FP_EX_OVERFLOW);		\ +    }							\ +  while (0) + +/* Check for a semi-raw value being a signaling NaN and raise the +   invalid exception if so.  */ +#define _FP_CHECK_SIGNAN_SEMIRAW(fs, wc, X)			\ +  do								\ +    {								\ +      if (X##_e == _FP_EXPMAX_##fs				\ +	  && !_FP_FRAC_ZEROP_##wc (X)				\ +	  && _FP_FRAC_SNANP_SEMIRAW (fs, X))			\ +	FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_SNAN);	\ +    }								\ +  while (0) + +/* Choose a NaN result from an operation on two semi-raw NaN +   values.  */ +#define _FP_CHOOSENAN_SEMIRAW(fs, wc, R, X, Y, OP)			\ +  do									\ +    {									\ +      /* _FP_CHOOSENAN expects raw values, so shift as required.  */	\ +      _FP_FRAC_SRL_##wc (X, _FP_WORKBITS);				\ +      _FP_FRAC_SRL_##wc (Y, _FP_WORKBITS);				\ +      _FP_CHOOSENAN (fs, wc, R, X, Y, OP);				\ +      _FP_FRAC_SLL_##wc (R, _FP_WORKBITS);				\ +    }									\ +  while (0) + +/* Make the fractional part a quiet NaN, preserving the payload +   if possible, otherwise make it the canonical quiet NaN and set +   the sign bit accordingly.  */ +#define _FP_SETQNAN(fs, wc, X)					\ +  do								\ +    {								\ +      if (_FP_QNANNEGATEDP)					\ +	{							\ +	  _FP_FRAC_HIGH_RAW_##fs (X) &= _FP_QNANBIT_##fs - 1;	\ +	  if (_FP_FRAC_ZEROP_##wc (X))				\ +	    {							\ +	      X##_s = _FP_NANSIGN_##fs;				\ +	      _FP_FRAC_SET_##wc (X, _FP_NANFRAC_##fs);		\ +	    }							\ +	}							\ +      else							\ +	_FP_FRAC_HIGH_RAW_##fs (X) |= _FP_QNANBIT_##fs;		\ +    }								\ +  while (0) +#define _FP_SETQNAN_SEMIRAW(fs, wc, X)				\ +  do								\ +    {								\ +      if (_FP_QNANNEGATEDP)					\ +	{							\ +	  _FP_FRAC_HIGH_##fs (X) &= _FP_QNANBIT_SH_##fs - 1;	\ +	  if (_FP_FRAC_ZEROP_##wc (X))				\ +	    {							\ +	      X##_s = _FP_NANSIGN_##fs;				\ +	      _FP_FRAC_SET_##wc (X, _FP_NANFRAC_##fs);		\ +	      _FP_FRAC_SLL_##wc (X, _FP_WORKBITS);		\ +	    }							\ +	}							\ +      else							\ +	_FP_FRAC_HIGH_##fs (X) |= _FP_QNANBIT_SH_##fs;		\ +    }								\ +  while (0) + +/* Test whether a biased exponent is normal (not zero or maximum).  */ +#define _FP_EXP_NORMAL(fs, wc, X)	(((X##_e + 1) & _FP_EXPMAX_##fs) > 1) + +/* Prepare to pack an fp value in semi-raw mode: the mantissa is +   rounded and shifted right, with the rounding possibly increasing +   the exponent (including changing a finite value to infinity).  */ +#define _FP_PACK_SEMIRAW(fs, wc, X)				\ +  do								\ +    {								\ +      int _FP_PACK_SEMIRAW_is_tiny				\ +	= X##_e == 0 && !_FP_FRAC_ZEROP_##wc (X);		\ +      if (_FP_TININESS_AFTER_ROUNDING				\ +	  && _FP_PACK_SEMIRAW_is_tiny)				\ +	{							\ +	  FP_DECL_##fs (_FP_PACK_SEMIRAW_T);			\ +	  _FP_FRAC_COPY_##wc (_FP_PACK_SEMIRAW_T, X);		\ +	  _FP_PACK_SEMIRAW_T##_s = X##_s;			\ +	  _FP_PACK_SEMIRAW_T##_e = X##_e;			\ +	  _FP_FRAC_SLL_##wc (_FP_PACK_SEMIRAW_T, 1);		\ +	  _FP_ROUND (wc, _FP_PACK_SEMIRAW_T);			\ +	  if (_FP_FRAC_OVERP_##wc (fs, _FP_PACK_SEMIRAW_T))	\ +	    _FP_PACK_SEMIRAW_is_tiny = 0;			\ +	}							\ +      _FP_ROUND (wc, X);					\ +      if (_FP_PACK_SEMIRAW_is_tiny)				\ +	{							\ +	  if ((FP_CUR_EXCEPTIONS & FP_EX_INEXACT)		\ +	      || (FP_TRAPPING_EXCEPTIONS & FP_EX_UNDERFLOW))	\ +	    FP_SET_EXCEPTION (FP_EX_UNDERFLOW);			\ +	}							\ +      if (_FP_FRAC_HIGH_##fs (X)				\ +	  & (_FP_OVERFLOW_##fs >> 1))				\ +	{							\ +	  _FP_FRAC_HIGH_##fs (X) &= ~(_FP_OVERFLOW_##fs >> 1);	\ +	  X##_e++;						\ +	  if (X##_e == _FP_EXPMAX_##fs)				\ +	    _FP_OVERFLOW_SEMIRAW (fs, wc, X);			\ +	}							\ +      _FP_FRAC_SRL_##wc (X, _FP_WORKBITS);			\ +      if (X##_e == _FP_EXPMAX_##fs && !_FP_FRAC_ZEROP_##wc (X))	\ +	{							\ +	  if (!_FP_KEEPNANFRACP)				\ +	    {							\ +	      _FP_FRAC_SET_##wc (X, _FP_NANFRAC_##fs);		\ +	      X##_s = _FP_NANSIGN_##fs;				\ +	    }							\ +	  else							\ +	    _FP_SETQNAN (fs, wc, X);				\ +	}							\ +    }								\ +  while (0) + +/* Before packing the bits back into the native fp result, take care +   of such mundane things as rounding and overflow.  Also, for some +   kinds of fp values, the original parts may not have been fully +   extracted -- but that is ok, we can regenerate them now.  */ + +#define _FP_PACK_CANONICAL(fs, wc, X)					\ +  do									\ +    {									\ +      switch (X##_c)							\ +	{								\ +	case FP_CLS_NORMAL:						\ +	  X##_e += _FP_EXPBIAS_##fs;					\ +	  if (X##_e > 0)						\ +	    {								\ +	      _FP_ROUND (wc, X);					\ +	      if (_FP_FRAC_OVERP_##wc (fs, X))				\ +		{							\ +		  _FP_FRAC_CLEAR_OVERP_##wc (fs, X);			\ +		  X##_e++;						\ +		}							\ +	      _FP_FRAC_SRL_##wc (X, _FP_WORKBITS);			\ +	      if (X##_e >= _FP_EXPMAX_##fs)				\ +		{							\ +		  /* Overflow.  */					\ +		  switch (FP_ROUNDMODE)					\ +		    {							\ +		    case FP_RND_NEAREST:				\ +		      X##_c = FP_CLS_INF;				\ +		      break;						\ +		    case FP_RND_PINF:					\ +		      if (!X##_s)					\ +			X##_c = FP_CLS_INF;				\ +		      break;						\ +		    case FP_RND_MINF:					\ +		      if (X##_s)					\ +			X##_c = FP_CLS_INF;				\ +		      break;						\ +		    }							\ +		  if (X##_c == FP_CLS_INF)				\ +		    {							\ +		      /* Overflow to infinity.  */			\ +		      X##_e = _FP_EXPMAX_##fs;				\ +		      _FP_FRAC_SET_##wc (X, _FP_ZEROFRAC_##wc);		\ +		    }							\ +		  else							\ +		    {							\ +		      /* Overflow to maximum normal.  */		\ +		      X##_e = _FP_EXPMAX_##fs - 1;			\ +		      _FP_FRAC_SET_##wc (X, _FP_MAXFRAC_##wc);		\ +		    }							\ +		  FP_SET_EXCEPTION (FP_EX_OVERFLOW);			\ +		  FP_SET_EXCEPTION (FP_EX_INEXACT);			\ +		}							\ +	    }								\ +	  else								\ +	    {								\ +	      /* We've got a denormalized number.  */			\ +	      int _FP_PACK_CANONICAL_is_tiny = 1;			\ +	      if (_FP_TININESS_AFTER_ROUNDING && X##_e == 0)		\ +		{							\ +		  FP_DECL_##fs (_FP_PACK_CANONICAL_T);			\ +		  _FP_FRAC_COPY_##wc (_FP_PACK_CANONICAL_T, X);		\ +		  _FP_PACK_CANONICAL_T##_s = X##_s;			\ +		  _FP_PACK_CANONICAL_T##_e = X##_e;			\ +		  _FP_ROUND (wc, _FP_PACK_CANONICAL_T);			\ +		  if (_FP_FRAC_OVERP_##wc (fs, _FP_PACK_CANONICAL_T))	\ +		    _FP_PACK_CANONICAL_is_tiny = 0;			\ +		}							\ +	      X##_e = -X##_e + 1;					\ +	      if (X##_e <= _FP_WFRACBITS_##fs)				\ +		{							\ +		  _FP_FRAC_SRS_##wc (X, X##_e, _FP_WFRACBITS_##fs);	\ +		  _FP_ROUND (wc, X);					\ +		  if (_FP_FRAC_HIGH_##fs (X)				\ +		      & (_FP_OVERFLOW_##fs >> 1))			\ +		    {							\ +		      X##_e = 1;					\ +		      _FP_FRAC_SET_##wc (X, _FP_ZEROFRAC_##wc);		\ +		      FP_SET_EXCEPTION (FP_EX_INEXACT);			\ +		    }							\ +		  else							\ +		    {							\ +		      X##_e = 0;					\ +		      _FP_FRAC_SRL_##wc (X, _FP_WORKBITS);		\ +		    }							\ +		  if (_FP_PACK_CANONICAL_is_tiny			\ +		      && ((FP_CUR_EXCEPTIONS & FP_EX_INEXACT)		\ +			  || (FP_TRAPPING_EXCEPTIONS			\ +			      & FP_EX_UNDERFLOW)))			\ +		    FP_SET_EXCEPTION (FP_EX_UNDERFLOW);			\ +		}							\ +	      else							\ +		{							\ +		  /* Underflow to zero.  */				\ +		  X##_e = 0;						\ +		  if (!_FP_FRAC_ZEROP_##wc (X))				\ +		    {							\ +		      _FP_FRAC_SET_##wc (X, _FP_MINFRAC_##wc);		\ +		      _FP_ROUND (wc, X);				\ +		      _FP_FRAC_LOW_##wc (X) >>= (_FP_WORKBITS);		\ +		    }							\ +		  FP_SET_EXCEPTION (FP_EX_UNDERFLOW);			\ +		}							\ +	    }								\ +	  break;							\ +									\ +	case FP_CLS_ZERO:						\ +	  X##_e = 0;							\ +	  _FP_FRAC_SET_##wc (X, _FP_ZEROFRAC_##wc);			\ +	  break;							\ +									\ +	case FP_CLS_INF:						\ +	  X##_e = _FP_EXPMAX_##fs;					\ +	  _FP_FRAC_SET_##wc (X, _FP_ZEROFRAC_##wc);			\ +	  break;							\ +									\ +	case FP_CLS_NAN:						\ +	  X##_e = _FP_EXPMAX_##fs;					\ +	  if (!_FP_KEEPNANFRACP)					\ +	    {								\ +	      _FP_FRAC_SET_##wc (X, _FP_NANFRAC_##fs);			\ +	      X##_s = _FP_NANSIGN_##fs;					\ +	    }								\ +	  else								\ +	    _FP_SETQNAN (fs, wc, X);					\ +	  break;							\ +	}								\ +    }									\ +  while (0) + +/* This one accepts raw argument and not cooked,  returns +   1 if X is a signaling NaN.  */ +#define _FP_ISSIGNAN(fs, wc, X)			\ +  ({						\ +    int _FP_ISSIGNAN_ret = 0;			\ +    if (X##_e == _FP_EXPMAX_##fs)		\ +      {						\ +	if (!_FP_FRAC_ZEROP_##wc (X)		\ +	    && _FP_FRAC_SNANP (fs, X))		\ +	  _FP_ISSIGNAN_ret = 1;			\ +      }						\ +    _FP_ISSIGNAN_ret;				\ +  }) + + + + + +/* Addition on semi-raw values.  */ +#define _FP_ADD_INTERNAL(fs, wc, R, X, Y, OP)				\ +  do									\ +    {									\ +      _FP_CHECK_FLUSH_ZERO (fs, wc, X);					\ +      _FP_CHECK_FLUSH_ZERO (fs, wc, Y);					\ +      if (X##_s == Y##_s)						\ +	{								\ +	  /* Addition.  */						\ +	  __label__ add1, add2, add3, add_done;				\ +	  R##_s = X##_s;						\ +	  int _FP_ADD_INTERNAL_ediff = X##_e - Y##_e;			\ +	  if (_FP_ADD_INTERNAL_ediff > 0)				\ +	    {								\ +	      R##_e = X##_e;						\ +	      if (Y##_e == 0)						\ +		{							\ +		  /* Y is zero or denormalized.  */			\ +		  if (_FP_FRAC_ZEROP_##wc (Y))				\ +		    {							\ +		      _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, X);		\ +		      _FP_FRAC_COPY_##wc (R, X);			\ +		      goto add_done;					\ +		    }							\ +		  else							\ +		    {							\ +		      FP_SET_EXCEPTION (FP_EX_DENORM);			\ +		      _FP_ADD_INTERNAL_ediff--;				\ +		      if (_FP_ADD_INTERNAL_ediff == 0)			\ +			{						\ +			  _FP_FRAC_ADD_##wc (R, X, Y);			\ +			  goto add3;					\ +			}						\ +		      if (X##_e == _FP_EXPMAX_##fs)			\ +			{						\ +			  _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, X);		\ +			  _FP_FRAC_COPY_##wc (R, X);			\ +			  goto add_done;				\ +			}						\ +		      goto add1;					\ +		    }							\ +		}							\ +	      else if (X##_e == _FP_EXPMAX_##fs)			\ +		{							\ +		  /* X is NaN or Inf, Y is normal.  */			\ +		  _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, X);			\ +		  _FP_FRAC_COPY_##wc (R, X);				\ +		  goto add_done;					\ +		}							\ +									\ +	      /* Insert implicit MSB of Y.  */				\ +	      _FP_FRAC_HIGH_##fs (Y) |= _FP_IMPLBIT_SH_##fs;		\ +									\ +	    add1:							\ +	      /* Shift the mantissa of Y to the right			\ +		 _FP_ADD_INTERNAL_EDIFF steps; remember to account	\ +		 later for the implicit MSB of X.  */			\ +	      if (_FP_ADD_INTERNAL_ediff <= _FP_WFRACBITS_##fs)		\ +		_FP_FRAC_SRS_##wc (Y, _FP_ADD_INTERNAL_ediff,		\ +				   _FP_WFRACBITS_##fs);			\ +	      else if (!_FP_FRAC_ZEROP_##wc (Y))			\ +		_FP_FRAC_SET_##wc (Y, _FP_MINFRAC_##wc);		\ +	      _FP_FRAC_ADD_##wc (R, X, Y);				\ +	    }								\ +	  else if (_FP_ADD_INTERNAL_ediff < 0)				\ +	    {								\ +	      _FP_ADD_INTERNAL_ediff = -_FP_ADD_INTERNAL_ediff;		\ +	      R##_e = Y##_e;						\ +	      if (X##_e == 0)						\ +		{							\ +		  /* X is zero or denormalized.  */			\ +		  if (_FP_FRAC_ZEROP_##wc (X))				\ +		    {							\ +		      _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, Y);		\ +		      _FP_FRAC_COPY_##wc (R, Y);			\ +		      goto add_done;					\ +		    }							\ +		  else							\ +		    {							\ +		      FP_SET_EXCEPTION (FP_EX_DENORM);			\ +		      _FP_ADD_INTERNAL_ediff--;				\ +		      if (_FP_ADD_INTERNAL_ediff == 0)			\ +			{						\ +			  _FP_FRAC_ADD_##wc (R, Y, X);			\ +			  goto add3;					\ +			}						\ +		      if (Y##_e == _FP_EXPMAX_##fs)			\ +			{						\ +			  _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, Y);		\ +			  _FP_FRAC_COPY_##wc (R, Y);			\ +			  goto add_done;				\ +			}						\ +		      goto add2;					\ +		    }							\ +		}							\ +	      else if (Y##_e == _FP_EXPMAX_##fs)			\ +		{							\ +		  /* Y is NaN or Inf, X is normal.  */			\ +		  _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, Y);			\ +		  _FP_FRAC_COPY_##wc (R, Y);				\ +		  goto add_done;					\ +		}							\ +									\ +	      /* Insert implicit MSB of X.  */				\ +	      _FP_FRAC_HIGH_##fs (X) |= _FP_IMPLBIT_SH_##fs;		\ +									\ +	    add2:							\ +	      /* Shift the mantissa of X to the right			\ +		 _FP_ADD_INTERNAL_EDIFF steps; remember to account	\ +		 later for the implicit MSB of Y.  */			\ +	      if (_FP_ADD_INTERNAL_ediff <= _FP_WFRACBITS_##fs)		\ +		_FP_FRAC_SRS_##wc (X, _FP_ADD_INTERNAL_ediff,		\ +				   _FP_WFRACBITS_##fs);			\ +	      else if (!_FP_FRAC_ZEROP_##wc (X))			\ +		_FP_FRAC_SET_##wc (X, _FP_MINFRAC_##wc);		\ +	      _FP_FRAC_ADD_##wc (R, Y, X);				\ +	    }								\ +	  else								\ +	    {								\ +	      /* _FP_ADD_INTERNAL_ediff == 0.  */			\ +	      if (!_FP_EXP_NORMAL (fs, wc, X))				\ +		{							\ +		  if (X##_e == 0)					\ +		    {							\ +		      /* X and Y are zero or denormalized.  */		\ +		      R##_e = 0;					\ +		      if (_FP_FRAC_ZEROP_##wc (X))			\ +			{						\ +			  if (!_FP_FRAC_ZEROP_##wc (Y))			\ +			    FP_SET_EXCEPTION (FP_EX_DENORM);		\ +			  _FP_FRAC_COPY_##wc (R, Y);			\ +			  goto add_done;				\ +			}						\ +		      else if (_FP_FRAC_ZEROP_##wc (Y))			\ +			{						\ +			  FP_SET_EXCEPTION (FP_EX_DENORM);		\ +			  _FP_FRAC_COPY_##wc (R, X);			\ +			  goto add_done;				\ +			}						\ +		      else						\ +			{						\ +			  FP_SET_EXCEPTION (FP_EX_DENORM);		\ +			  _FP_FRAC_ADD_##wc (R, X, Y);			\ +			  if (_FP_FRAC_HIGH_##fs (R) & _FP_IMPLBIT_SH_##fs) \ +			    {						\ +			      /* Normalized result.  */			\ +			      _FP_FRAC_HIGH_##fs (R)			\ +				&= ~(_FP_W_TYPE) _FP_IMPLBIT_SH_##fs;	\ +			      R##_e = 1;				\ +			    }						\ +			  goto add_done;				\ +			}						\ +		    }							\ +		  else							\ +		    {							\ +		      /* X and Y are NaN or Inf.  */			\ +		      _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, X);		\ +		      _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, Y);		\ +		      R##_e = _FP_EXPMAX_##fs;				\ +		      if (_FP_FRAC_ZEROP_##wc (X))			\ +			_FP_FRAC_COPY_##wc (R, Y);			\ +		      else if (_FP_FRAC_ZEROP_##wc (Y))			\ +			_FP_FRAC_COPY_##wc (R, X);			\ +		      else						\ +			_FP_CHOOSENAN_SEMIRAW (fs, wc, R, X, Y, OP);	\ +		      goto add_done;					\ +		    }							\ +		}							\ +	      /* The exponents of X and Y, both normal, are equal.  The	\ +		 implicit MSBs will always add to increase the		\ +		 exponent.  */						\ +	      _FP_FRAC_ADD_##wc (R, X, Y);				\ +	      R##_e = X##_e + 1;					\ +	      _FP_FRAC_SRS_##wc (R, 1, _FP_WFRACBITS_##fs);		\ +	      if (R##_e == _FP_EXPMAX_##fs)				\ +		/* Overflow to infinity (depending on rounding mode).  */ \ +		_FP_OVERFLOW_SEMIRAW (fs, wc, R);			\ +	      goto add_done;						\ +	    }								\ +	add3:								\ +	  if (_FP_FRAC_HIGH_##fs (R) & _FP_IMPLBIT_SH_##fs)		\ +	    {								\ +	      /* Overflow.  */						\ +	      _FP_FRAC_HIGH_##fs (R) &= ~(_FP_W_TYPE) _FP_IMPLBIT_SH_##fs; \ +	      R##_e++;							\ +	      _FP_FRAC_SRS_##wc (R, 1, _FP_WFRACBITS_##fs);		\ +	      if (R##_e == _FP_EXPMAX_##fs)				\ +		/* Overflow to infinity (depending on rounding mode).  */ \ +		_FP_OVERFLOW_SEMIRAW (fs, wc, R);			\ +	    }								\ +	add_done: ;							\ +	}								\ +      else								\ +	{								\ +	  /* Subtraction.  */						\ +	  __label__ sub1, sub2, sub3, norm, sub_done;			\ +	  int _FP_ADD_INTERNAL_ediff = X##_e - Y##_e;			\ +	  if (_FP_ADD_INTERNAL_ediff > 0)				\ +	    {								\ +	      R##_e = X##_e;						\ +	      R##_s = X##_s;						\ +	      if (Y##_e == 0)						\ +		{							\ +		  /* Y is zero or denormalized.  */			\ +		  if (_FP_FRAC_ZEROP_##wc (Y))				\ +		    {							\ +		      _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, X);		\ +		      _FP_FRAC_COPY_##wc (R, X);			\ +		      goto sub_done;					\ +		    }							\ +		  else							\ +		    {							\ +		      FP_SET_EXCEPTION (FP_EX_DENORM);			\ +		      _FP_ADD_INTERNAL_ediff--;				\ +		      if (_FP_ADD_INTERNAL_ediff == 0)			\ +			{						\ +			  _FP_FRAC_SUB_##wc (R, X, Y);			\ +			  goto sub3;					\ +			}						\ +		      if (X##_e == _FP_EXPMAX_##fs)			\ +			{						\ +			  _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, X);		\ +			  _FP_FRAC_COPY_##wc (R, X);			\ +			  goto sub_done;				\ +			}						\ +		      goto sub1;					\ +		    }							\ +		}							\ +	      else if (X##_e == _FP_EXPMAX_##fs)			\ +		{							\ +		  /* X is NaN or Inf, Y is normal.  */			\ +		  _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, X);			\ +		  _FP_FRAC_COPY_##wc (R, X);				\ +		  goto sub_done;					\ +		}							\ +									\ +	      /* Insert implicit MSB of Y.  */				\ +	      _FP_FRAC_HIGH_##fs (Y) |= _FP_IMPLBIT_SH_##fs;		\ +									\ +	    sub1:							\ +	      /* Shift the mantissa of Y to the right			\ +		 _FP_ADD_INTERNAL_EDIFF steps; remember to account	\ +		 later for the implicit MSB of X.  */			\ +	      if (_FP_ADD_INTERNAL_ediff <= _FP_WFRACBITS_##fs)		\ +		_FP_FRAC_SRS_##wc (Y, _FP_ADD_INTERNAL_ediff,		\ +				   _FP_WFRACBITS_##fs);			\ +	      else if (!_FP_FRAC_ZEROP_##wc (Y))			\ +		_FP_FRAC_SET_##wc (Y, _FP_MINFRAC_##wc);		\ +	      _FP_FRAC_SUB_##wc (R, X, Y);				\ +	    }								\ +	  else if (_FP_ADD_INTERNAL_ediff < 0)				\ +	    {								\ +	      _FP_ADD_INTERNAL_ediff = -_FP_ADD_INTERNAL_ediff;		\ +	      R##_e = Y##_e;						\ +	      R##_s = Y##_s;						\ +	      if (X##_e == 0)						\ +		{							\ +		  /* X is zero or denormalized.  */			\ +		  if (_FP_FRAC_ZEROP_##wc (X))				\ +		    {							\ +		      _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, Y);		\ +		      _FP_FRAC_COPY_##wc (R, Y);			\ +		      goto sub_done;					\ +		    }							\ +		  else							\ +		    {							\ +		      FP_SET_EXCEPTION (FP_EX_DENORM);			\ +		      _FP_ADD_INTERNAL_ediff--;				\ +		      if (_FP_ADD_INTERNAL_ediff == 0)			\ +			{						\ +			  _FP_FRAC_SUB_##wc (R, Y, X);			\ +			  goto sub3;					\ +			}						\ +		      if (Y##_e == _FP_EXPMAX_##fs)			\ +			{						\ +			  _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, Y);		\ +			  _FP_FRAC_COPY_##wc (R, Y);			\ +			  goto sub_done;				\ +			}						\ +		      goto sub2;					\ +		    }							\ +		}							\ +	      else if (Y##_e == _FP_EXPMAX_##fs)			\ +		{							\ +		  /* Y is NaN or Inf, X is normal.  */			\ +		  _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, Y);			\ +		  _FP_FRAC_COPY_##wc (R, Y);				\ +		  goto sub_done;					\ +		}							\ +									\ +	      /* Insert implicit MSB of X.  */				\ +	      _FP_FRAC_HIGH_##fs (X) |= _FP_IMPLBIT_SH_##fs;		\ +									\ +	    sub2:							\ +	      /* Shift the mantissa of X to the right			\ +		 _FP_ADD_INTERNAL_EDIFF steps; remember to account	\ +		 later for the implicit MSB of Y.  */			\ +	      if (_FP_ADD_INTERNAL_ediff <= _FP_WFRACBITS_##fs)		\ +		_FP_FRAC_SRS_##wc (X, _FP_ADD_INTERNAL_ediff,		\ +				   _FP_WFRACBITS_##fs);			\ +	      else if (!_FP_FRAC_ZEROP_##wc (X))			\ +		_FP_FRAC_SET_##wc (X, _FP_MINFRAC_##wc);		\ +	      _FP_FRAC_SUB_##wc (R, Y, X);				\ +	    }								\ +	  else								\ +	    {								\ +	      /* ediff == 0.  */					\ +	      if (!_FP_EXP_NORMAL (fs, wc, X))				\ +		{							\ +		  if (X##_e == 0)					\ +		    {							\ +		      /* X and Y are zero or denormalized.  */		\ +		      R##_e = 0;					\ +		      if (_FP_FRAC_ZEROP_##wc (X))			\ +			{						\ +			  _FP_FRAC_COPY_##wc (R, Y);			\ +			  if (_FP_FRAC_ZEROP_##wc (Y))			\ +			    R##_s = (FP_ROUNDMODE == FP_RND_MINF);	\ +			  else						\ +			    {						\ +			      FP_SET_EXCEPTION (FP_EX_DENORM);		\ +			      R##_s = Y##_s;				\ +			    }						\ +			  goto sub_done;				\ +			}						\ +		      else if (_FP_FRAC_ZEROP_##wc (Y))			\ +			{						\ +			  FP_SET_EXCEPTION (FP_EX_DENORM);		\ +			  _FP_FRAC_COPY_##wc (R, X);			\ +			  R##_s = X##_s;				\ +			  goto sub_done;				\ +			}						\ +		      else						\ +			{						\ +			  FP_SET_EXCEPTION (FP_EX_DENORM);		\ +			  _FP_FRAC_SUB_##wc (R, X, Y);			\ +			  R##_s = X##_s;				\ +			  if (_FP_FRAC_HIGH_##fs (R) & _FP_IMPLBIT_SH_##fs) \ +			    {						\ +			      /* |X| < |Y|, negate result.  */		\ +			      _FP_FRAC_SUB_##wc (R, Y, X);		\ +			      R##_s = Y##_s;				\ +			    }						\ +			  else if (_FP_FRAC_ZEROP_##wc (R))		\ +			    R##_s = (FP_ROUNDMODE == FP_RND_MINF);	\ +			  goto sub_done;				\ +			}						\ +		    }							\ +		  else							\ +		    {							\ +		      /* X and Y are NaN or Inf, of opposite signs.  */	\ +		      _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, X);		\ +		      _FP_CHECK_SIGNAN_SEMIRAW (fs, wc, Y);		\ +		      R##_e = _FP_EXPMAX_##fs;				\ +		      if (_FP_FRAC_ZEROP_##wc (X))			\ +			{						\ +			  if (_FP_FRAC_ZEROP_##wc (Y))			\ +			    {						\ +			      /* Inf - Inf.  */				\ +			      R##_s = _FP_NANSIGN_##fs;			\ +			      _FP_FRAC_SET_##wc (R, _FP_NANFRAC_##fs);	\ +			      _FP_FRAC_SLL_##wc (R, _FP_WORKBITS);	\ +			      FP_SET_EXCEPTION (FP_EX_INVALID		\ +						| FP_EX_INVALID_ISI);	\ +			    }						\ +			  else						\ +			    {						\ +			      /* Inf - NaN.  */				\ +			      R##_s = Y##_s;				\ +			      _FP_FRAC_COPY_##wc (R, Y);		\ +			    }						\ +			}						\ +		      else						\ +			{						\ +			  if (_FP_FRAC_ZEROP_##wc (Y))			\ +			    {						\ +			      /* NaN - Inf.  */				\ +			      R##_s = X##_s;				\ +			      _FP_FRAC_COPY_##wc (R, X);		\ +			    }						\ +			  else						\ +			    {						\ +			      /* NaN - NaN.  */				\ +			      _FP_CHOOSENAN_SEMIRAW (fs, wc, R, X, Y, OP); \ +			    }						\ +			}						\ +		      goto sub_done;					\ +		    }							\ +		}							\ +	      /* The exponents of X and Y, both normal, are equal.  The	\ +		 implicit MSBs cancel.  */				\ +	      R##_e = X##_e;						\ +	      _FP_FRAC_SUB_##wc (R, X, Y);				\ +	      R##_s = X##_s;						\ +	      if (_FP_FRAC_HIGH_##fs (R) & _FP_IMPLBIT_SH_##fs)		\ +		{							\ +		  /* |X| < |Y|, negate result.  */			\ +		  _FP_FRAC_SUB_##wc (R, Y, X);				\ +		  R##_s = Y##_s;					\ +		}							\ +	      else if (_FP_FRAC_ZEROP_##wc (R))				\ +		{							\ +		  R##_e = 0;						\ +		  R##_s = (FP_ROUNDMODE == FP_RND_MINF);		\ +		  goto sub_done;					\ +		}							\ +	      goto norm;						\ +	    }								\ +	sub3:								\ +	  if (_FP_FRAC_HIGH_##fs (R) & _FP_IMPLBIT_SH_##fs)		\ +	    {								\ +	      int _FP_ADD_INTERNAL_diff;				\ +	      /* Carry into most significant bit of larger one of X and Y, \ +		 canceling it; renormalize.  */				\ +	      _FP_FRAC_HIGH_##fs (R) &= _FP_IMPLBIT_SH_##fs - 1;	\ +	    norm:							\ +	      _FP_FRAC_CLZ_##wc (_FP_ADD_INTERNAL_diff, R);		\ +	      _FP_ADD_INTERNAL_diff -= _FP_WFRACXBITS_##fs;		\ +	      _FP_FRAC_SLL_##wc (R, _FP_ADD_INTERNAL_diff);		\ +	      if (R##_e <= _FP_ADD_INTERNAL_diff)			\ +		{							\ +		  /* R is denormalized.  */				\ +		  _FP_ADD_INTERNAL_diff					\ +		    = _FP_ADD_INTERNAL_diff - R##_e + 1;		\ +		  _FP_FRAC_SRS_##wc (R, _FP_ADD_INTERNAL_diff,		\ +				     _FP_WFRACBITS_##fs);		\ +		  R##_e = 0;						\ +		}							\ +	      else							\ +		{							\ +		  R##_e -= _FP_ADD_INTERNAL_diff;			\ +		  _FP_FRAC_HIGH_##fs (R) &= ~(_FP_W_TYPE) _FP_IMPLBIT_SH_##fs; \ +		}							\ +	    }								\ +	sub_done: ;							\ +	}								\ +    }									\ +  while (0) + +#define _FP_ADD(fs, wc, R, X, Y) _FP_ADD_INTERNAL (fs, wc, R, X, Y, '+') +#define _FP_SUB(fs, wc, R, X, Y)					\ +  do									\ +    {									\ +      if (!(Y##_e == _FP_EXPMAX_##fs && !_FP_FRAC_ZEROP_##wc (Y)))	\ +	Y##_s ^= 1;							\ +      _FP_ADD_INTERNAL (fs, wc, R, X, Y, '-');				\ +    }									\ +  while (0) + + +/* Main negation routine.  The input value is raw.  */ + +#define _FP_NEG(fs, wc, R, X)			\ +  do						\ +    {						\ +      _FP_FRAC_COPY_##wc (R, X);		\ +      R##_e = X##_e;				\ +      R##_s = 1 ^ X##_s;			\ +    }						\ +  while (0) + + +/* Main multiplication routine.  The input values should be cooked.  */ + +#define _FP_MUL(fs, wc, R, X, Y)				\ +  do								\ +    {								\ +      R##_s = X##_s ^ Y##_s;					\ +      R##_e = X##_e + Y##_e + 1;				\ +      switch (_FP_CLS_COMBINE (X##_c, Y##_c))			\ +	{							\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_NORMAL):	\ +	  R##_c = FP_CLS_NORMAL;				\ +								\ +	  _FP_MUL_MEAT_##fs (R, X, Y);				\ +								\ +	  if (_FP_FRAC_OVERP_##wc (fs, R))			\ +	    _FP_FRAC_SRS_##wc (R, 1, _FP_WFRACBITS_##fs);	\ +	  else							\ +	    R##_e--;						\ +	  break;						\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_NAN):		\ +	  _FP_CHOOSENAN (fs, wc, R, X, Y, '*');			\ +	  break;						\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_NORMAL):	\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_INF):		\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_ZERO):		\ +	  R##_s = X##_s;					\ +	  /* FALLTHRU */					\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_INF):		\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_NORMAL):	\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_NORMAL):	\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_ZERO):	\ +	  _FP_FRAC_COPY_##wc (R, X);				\ +	  R##_c = X##_c;					\ +	  break;						\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_NAN):	\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_NAN):		\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_NAN):		\ +	  R##_s = Y##_s;					\ +	  /* FALLTHRU */					\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_INF):	\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_ZERO):	\ +	  _FP_FRAC_COPY_##wc (R, Y);				\ +	  R##_c = Y##_c;					\ +	  break;						\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_ZERO):		\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_INF):		\ +	  R##_s = _FP_NANSIGN_##fs;				\ +	  R##_c = FP_CLS_NAN;					\ +	  _FP_FRAC_SET_##wc (R, _FP_NANFRAC_##fs);		\ +	  FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_IMZ);	\ +	  break;						\ +								\ +	default:						\ +	  _FP_UNREACHABLE;					\ +	}							\ +    }								\ +  while (0) + + +/* Fused multiply-add.  The input values should be cooked.  */ + +#define _FP_FMA(fs, wc, dwc, R, X, Y, Z)				\ +  do									\ +    {									\ +      __label__ done_fma;						\ +      FP_DECL_##fs (_FP_FMA_T);						\ +      _FP_FMA_T##_s = X##_s ^ Y##_s;					\ +      _FP_FMA_T##_e = X##_e + Y##_e + 1;				\ +      switch (_FP_CLS_COMBINE (X##_c, Y##_c))				\ +	{								\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_NORMAL):		\ +	  switch (Z##_c)						\ +	    {								\ +	    case FP_CLS_INF:						\ +	    case FP_CLS_NAN:						\ +	      R##_s = Z##_s;						\ +	      _FP_FRAC_COPY_##wc (R, Z);				\ +	      R##_c = Z##_c;						\ +	      break;							\ +									\ +	    case FP_CLS_ZERO:						\ +	      R##_c = FP_CLS_NORMAL;					\ +	      R##_s = _FP_FMA_T##_s;					\ +	      R##_e = _FP_FMA_T##_e;					\ +									\ +	      _FP_MUL_MEAT_##fs (R, X, Y);				\ +									\ +	      if (_FP_FRAC_OVERP_##wc (fs, R))				\ +		_FP_FRAC_SRS_##wc (R, 1, _FP_WFRACBITS_##fs);		\ +	      else							\ +		R##_e--;						\ +	      break;							\ +									\ +	    case FP_CLS_NORMAL:;					\ +	      _FP_FRAC_DECL_##dwc (_FP_FMA_TD);				\ +	      _FP_FRAC_DECL_##dwc (_FP_FMA_ZD);				\ +	      _FP_FRAC_DECL_##dwc (_FP_FMA_RD);				\ +	      _FP_MUL_MEAT_DW_##fs (_FP_FMA_TD, X, Y);			\ +	      R##_e = _FP_FMA_T##_e;					\ +	      int _FP_FMA_tsh						\ +		= _FP_FRAC_HIGHBIT_DW_##dwc (fs, _FP_FMA_TD) == 0;	\ +	      _FP_FMA_T##_e -= _FP_FMA_tsh;				\ +	      int _FP_FMA_ediff = _FP_FMA_T##_e - Z##_e;		\ +	      if (_FP_FMA_ediff >= 0)					\ +		{							\ +		  int _FP_FMA_shift					\ +		    = _FP_WFRACBITS_##fs - _FP_FMA_tsh - _FP_FMA_ediff;	\ +		  if (_FP_FMA_shift <= -_FP_WFRACBITS_##fs)		\ +		    _FP_FRAC_SET_##dwc (_FP_FMA_ZD, _FP_MINFRAC_##dwc);	\ +		  else							\ +		    {							\ +		      _FP_FRAC_COPY_##dwc##_##wc (_FP_FMA_ZD, Z);	\ +		      if (_FP_FMA_shift < 0)				\ +			_FP_FRAC_SRS_##dwc (_FP_FMA_ZD, -_FP_FMA_shift,	\ +					    _FP_WFRACBITS_DW_##fs);	\ +		      else if (_FP_FMA_shift > 0)			\ +			_FP_FRAC_SLL_##dwc (_FP_FMA_ZD, _FP_FMA_shift);	\ +		    }							\ +		  R##_s = _FP_FMA_T##_s;				\ +		  if (_FP_FMA_T##_s == Z##_s)				\ +		    _FP_FRAC_ADD_##dwc (_FP_FMA_RD, _FP_FMA_TD,		\ +					_FP_FMA_ZD);			\ +		  else							\ +		    {							\ +		      _FP_FRAC_SUB_##dwc (_FP_FMA_RD, _FP_FMA_TD,	\ +					  _FP_FMA_ZD);			\ +		      if (_FP_FRAC_NEGP_##dwc (_FP_FMA_RD))		\ +			{						\ +			  R##_s = Z##_s;				\ +			  _FP_FRAC_SUB_##dwc (_FP_FMA_RD, _FP_FMA_ZD,	\ +					      _FP_FMA_TD);		\ +			}						\ +		    }							\ +		}							\ +	      else							\ +		{							\ +		  R##_e = Z##_e;					\ +		  R##_s = Z##_s;					\ +		  _FP_FRAC_COPY_##dwc##_##wc (_FP_FMA_ZD, Z);		\ +		  _FP_FRAC_SLL_##dwc (_FP_FMA_ZD, _FP_WFRACBITS_##fs);	\ +		  int _FP_FMA_shift = -_FP_FMA_ediff - _FP_FMA_tsh;	\ +		  if (_FP_FMA_shift >= _FP_WFRACBITS_DW_##fs)		\ +		    _FP_FRAC_SET_##dwc (_FP_FMA_TD, _FP_MINFRAC_##dwc);	\ +		  else if (_FP_FMA_shift > 0)				\ +		    _FP_FRAC_SRS_##dwc (_FP_FMA_TD, _FP_FMA_shift,	\ +					_FP_WFRACBITS_DW_##fs);		\ +		  if (Z##_s == _FP_FMA_T##_s)				\ +		    _FP_FRAC_ADD_##dwc (_FP_FMA_RD, _FP_FMA_ZD,		\ +					_FP_FMA_TD);			\ +		  else							\ +		    _FP_FRAC_SUB_##dwc (_FP_FMA_RD, _FP_FMA_ZD,		\ +					_FP_FMA_TD);			\ +		}							\ +	      if (_FP_FRAC_ZEROP_##dwc (_FP_FMA_RD))			\ +		{							\ +		  if (_FP_FMA_T##_s == Z##_s)				\ +		    R##_s = Z##_s;					\ +		  else							\ +		    R##_s = (FP_ROUNDMODE == FP_RND_MINF);		\ +		  _FP_FRAC_SET_##wc (R, _FP_ZEROFRAC_##wc);		\ +		  R##_c = FP_CLS_ZERO;					\ +		}							\ +	      else							\ +		{							\ +		  int _FP_FMA_rlz;					\ +		  _FP_FRAC_CLZ_##dwc (_FP_FMA_rlz, _FP_FMA_RD);		\ +		  _FP_FMA_rlz -= _FP_WFRACXBITS_DW_##fs;		\ +		  R##_e -= _FP_FMA_rlz;					\ +		  int _FP_FMA_shift = _FP_WFRACBITS_##fs - _FP_FMA_rlz;	\ +		  if (_FP_FMA_shift > 0)				\ +		    _FP_FRAC_SRS_##dwc (_FP_FMA_RD, _FP_FMA_shift,	\ +					_FP_WFRACBITS_DW_##fs);		\ +		  else if (_FP_FMA_shift < 0)				\ +		    _FP_FRAC_SLL_##dwc (_FP_FMA_RD, -_FP_FMA_shift);	\ +		  _FP_FRAC_COPY_##wc##_##dwc (R, _FP_FMA_RD);		\ +		  R##_c = FP_CLS_NORMAL;				\ +		}							\ +	      break;							\ +	    }								\ +	  goto done_fma;						\ +									\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_NAN):			\ +	  _FP_CHOOSENAN (fs, wc, _FP_FMA_T, X, Y, '*');			\ +	  break;							\ +									\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_NORMAL):		\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_INF):			\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_ZERO):			\ +	  _FP_FMA_T##_s = X##_s;					\ +	  /* FALLTHRU */						\ +									\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_INF):			\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_NORMAL):		\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_NORMAL):		\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_ZERO):		\ +	  _FP_FRAC_COPY_##wc (_FP_FMA_T, X);				\ +	  _FP_FMA_T##_c = X##_c;					\ +	  break;							\ +									\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_NAN):		\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_NAN):			\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_NAN):			\ +	  _FP_FMA_T##_s = Y##_s;					\ +	  /* FALLTHRU */						\ +									\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_INF):		\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_ZERO):		\ +	  _FP_FRAC_COPY_##wc (_FP_FMA_T, Y);				\ +	  _FP_FMA_T##_c = Y##_c;					\ +	  break;							\ +									\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_ZERO):			\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_INF):			\ +	  _FP_FMA_T##_s = _FP_NANSIGN_##fs;				\ +	  _FP_FMA_T##_c = FP_CLS_NAN;					\ +	  _FP_FRAC_SET_##wc (_FP_FMA_T, _FP_NANFRAC_##fs);		\ +	  FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_IMZ_FMA);	\ +	  break;							\ +									\ +	default:							\ +	  _FP_UNREACHABLE;						\ +	}								\ +									\ +      /* T = X * Y is zero, infinity or NaN.  */			\ +      switch (_FP_CLS_COMBINE (_FP_FMA_T##_c, Z##_c))			\ +	{								\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_NAN):			\ +	  _FP_CHOOSENAN (fs, wc, R, _FP_FMA_T, Z, '+');			\ +	  break;							\ +									\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_NORMAL):		\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_INF):			\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_ZERO):			\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_NORMAL):		\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_ZERO):			\ +	  R##_s = _FP_FMA_T##_s;					\ +	  _FP_FRAC_COPY_##wc (R, _FP_FMA_T);				\ +	  R##_c = _FP_FMA_T##_c;					\ +	  break;							\ +									\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_NAN):			\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_NAN):			\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_NORMAL):		\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_INF):			\ +	  R##_s = Z##_s;						\ +	  _FP_FRAC_COPY_##wc (R, Z);					\ +	  R##_c = Z##_c;						\ +	  R##_e = Z##_e;						\ +	  break;							\ +									\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_INF):			\ +	  if (_FP_FMA_T##_s == Z##_s)					\ +	    {								\ +	      R##_s = Z##_s;						\ +	      _FP_FRAC_COPY_##wc (R, Z);				\ +	      R##_c = Z##_c;						\ +	    }								\ +	  else								\ +	    {								\ +	      R##_s = _FP_NANSIGN_##fs;					\ +	      R##_c = FP_CLS_NAN;					\ +	      _FP_FRAC_SET_##wc (R, _FP_NANFRAC_##fs);			\ +	      FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_ISI);	\ +	    }								\ +	  break;							\ +									\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_ZERO):		\ +	  if (_FP_FMA_T##_s == Z##_s)					\ +	    R##_s = Z##_s;						\ +	  else								\ +	    R##_s = (FP_ROUNDMODE == FP_RND_MINF);			\ +	  _FP_FRAC_COPY_##wc (R, Z);					\ +	  R##_c = Z##_c;						\ +	  break;							\ +									\ +	default:							\ +	  _FP_UNREACHABLE;						\ +	}								\ +    done_fma: ;								\ +    }									\ +  while (0) + + +/* Main division routine.  The input values should be cooked.  */ + +#define _FP_DIV(fs, wc, R, X, Y)				\ +  do								\ +    {								\ +      R##_s = X##_s ^ Y##_s;					\ +      R##_e = X##_e - Y##_e;					\ +      switch (_FP_CLS_COMBINE (X##_c, Y##_c))			\ +	{							\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_NORMAL):	\ +	  R##_c = FP_CLS_NORMAL;				\ +								\ +	  _FP_DIV_MEAT_##fs (R, X, Y);				\ +	  break;						\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_NAN):		\ +	  _FP_CHOOSENAN (fs, wc, R, X, Y, '/');			\ +	  break;						\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_NORMAL):	\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_INF):		\ +	case _FP_CLS_COMBINE (FP_CLS_NAN, FP_CLS_ZERO):		\ +	  R##_s = X##_s;					\ +	  _FP_FRAC_COPY_##wc (R, X);				\ +	  R##_c = X##_c;					\ +	  break;						\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_NAN):	\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_NAN):		\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_NAN):		\ +	  R##_s = Y##_s;					\ +	  _FP_FRAC_COPY_##wc (R, Y);				\ +	  R##_c = Y##_c;					\ +	  break;						\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_INF):	\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_INF):		\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_NORMAL):	\ +	  R##_c = FP_CLS_ZERO;					\ +	  break;						\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_NORMAL, FP_CLS_ZERO):	\ +	  FP_SET_EXCEPTION (FP_EX_DIVZERO);			\ +	  /* FALLTHRU */					\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_ZERO):		\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_NORMAL):	\ +	  R##_c = FP_CLS_INF;					\ +	  break;						\ +								\ +	case _FP_CLS_COMBINE (FP_CLS_INF, FP_CLS_INF):		\ +	case _FP_CLS_COMBINE (FP_CLS_ZERO, FP_CLS_ZERO):	\ +	  R##_s = _FP_NANSIGN_##fs;				\ +	  R##_c = FP_CLS_NAN;					\ +	  _FP_FRAC_SET_##wc (R, _FP_NANFRAC_##fs);		\ +	  FP_SET_EXCEPTION (FP_EX_INVALID			\ +			    | (X##_c == FP_CLS_INF		\ +			       ? FP_EX_INVALID_IDI		\ +			       : FP_EX_INVALID_ZDZ));		\ +	  break;						\ +								\ +	default:						\ +	  _FP_UNREACHABLE;					\ +	}							\ +    }								\ +  while (0) + + +/* Helper for comparisons.  EX is 0 not to raise exceptions, 1 to +   raise exceptions for signaling NaN operands, 2 to raise exceptions +   for all NaN operands.  Conditionals are organized to allow the +   compiler to optimize away code based on the value of EX.  */ + +#define _FP_CMP_CHECK_NAN(fs, wc, X, Y, ex)				\ +  do									\ +    {									\ +      /* The arguments are unordered, which may or may not result in	\ +	 an exception.  */						\ +      if (ex)								\ +	{								\ +	  /* At least some cases of unordered arguments result in	\ +	     exceptions; check whether this is one.  */			\ +	  if (FP_EX_INVALID_SNAN || FP_EX_INVALID_VC)			\ +	    {								\ +	      /* Check separately for each case of "invalid"		\ +		 exceptions.  */					\ +	      if ((ex) == 2)						\ +		FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_VC);	\ +	      if (_FP_ISSIGNAN (fs, wc, X)				\ +		  || _FP_ISSIGNAN (fs, wc, Y))				\ +		FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_SNAN);	\ +	    }								\ +	  /* Otherwise, we only need to check whether to raise an	\ +	     exception, not which case or cases it is.  */		\ +	  else if ((ex) == 2						\ +		   || _FP_ISSIGNAN (fs, wc, X)				\ +		   || _FP_ISSIGNAN (fs, wc, Y))				\ +	    FP_SET_EXCEPTION (FP_EX_INVALID);				\ +	}								\ +    }									\ +  while (0) + +/* Helper for comparisons.  If denormal operands would raise an +   exception, check for them, and flush to zero as appropriate +   (otherwise, we need only check and flush to zero if it might affect +   the result, which is done later with _FP_CMP_CHECK_FLUSH_ZERO).  */ +#define _FP_CMP_CHECK_DENORM(fs, wc, X, Y)				\ +  do									\ +    {									\ +      if (FP_EX_DENORM != 0)						\ +	{								\ +	  /* We must ensure the correct exceptions are raised for	\ +	     denormal operands, even though this may not affect the	\ +	     result of the comparison.  */				\ +	  if (FP_DENORM_ZERO)						\ +	    {								\ +	      _FP_CHECK_FLUSH_ZERO (fs, wc, X);				\ +	      _FP_CHECK_FLUSH_ZERO (fs, wc, Y);				\ +	    }								\ +	  else								\ +	    {								\ +	      if ((X##_e == 0 && !_FP_FRAC_ZEROP_##wc (X))		\ +		  || (Y##_e == 0 && !_FP_FRAC_ZEROP_##wc (Y)))		\ +		FP_SET_EXCEPTION (FP_EX_DENORM);			\ +	    }								\ +	}								\ +    }									\ +  while (0) + +/* Helper for comparisons.  Check for flushing denormals for zero if +   we didn't need to check earlier for any denormal operands.  */ +#define _FP_CMP_CHECK_FLUSH_ZERO(fs, wc, X, Y)	\ +  do						\ +    {						\ +      if (FP_EX_DENORM == 0)			\ +	{					\ +	  _FP_CHECK_FLUSH_ZERO (fs, wc, X);	\ +	  _FP_CHECK_FLUSH_ZERO (fs, wc, Y);	\ +	}					\ +    }						\ +  while (0) + +/* Main differential comparison routine.  The inputs should be raw not +   cooked.  The return is -1, 0, 1 for normal values, UN +   otherwise.  */ + +#define _FP_CMP(fs, wc, ret, X, Y, un, ex)				\ +  do									\ +    {									\ +      _FP_CMP_CHECK_DENORM (fs, wc, X, Y);				\ +      /* NANs are unordered.  */					\ +      if ((X##_e == _FP_EXPMAX_##fs && !_FP_FRAC_ZEROP_##wc (X))	\ +	  || (Y##_e == _FP_EXPMAX_##fs && !_FP_FRAC_ZEROP_##wc (Y)))	\ +	{								\ +	  (ret) = (un);							\ +	  _FP_CMP_CHECK_NAN (fs, wc, X, Y, (ex));			\ +	}								\ +      else								\ +	{								\ +	  int _FP_CMP_is_zero_x;					\ +	  int _FP_CMP_is_zero_y;					\ +									\ +	  _FP_CMP_CHECK_FLUSH_ZERO (fs, wc, X, Y);			\ +									\ +	  _FP_CMP_is_zero_x						\ +	    = (!X##_e && _FP_FRAC_ZEROP_##wc (X)) ? 1 : 0;		\ +	  _FP_CMP_is_zero_y						\ +	    = (!Y##_e && _FP_FRAC_ZEROP_##wc (Y)) ? 1 : 0;		\ +									\ +	  if (_FP_CMP_is_zero_x && _FP_CMP_is_zero_y)			\ +	    (ret) = 0;							\ +	  else if (_FP_CMP_is_zero_x)					\ +	    (ret) = Y##_s ? 1 : -1;					\ +	  else if (_FP_CMP_is_zero_y)					\ +	    (ret) = X##_s ? -1 : 1;					\ +	  else if (X##_s != Y##_s)					\ +	    (ret) = X##_s ? -1 : 1;					\ +	  else if (X##_e > Y##_e)					\ +	    (ret) = X##_s ? -1 : 1;					\ +	  else if (X##_e < Y##_e)					\ +	    (ret) = X##_s ? 1 : -1;					\ +	  else if (_FP_FRAC_GT_##wc (X, Y))				\ +	    (ret) = X##_s ? -1 : 1;					\ +	  else if (_FP_FRAC_GT_##wc (Y, X))				\ +	    (ret) = X##_s ? 1 : -1;					\ +	  else								\ +	    (ret) = 0;							\ +	}								\ +    }									\ +  while (0) + + +/* Simplification for strict equality.  */ + +#define _FP_CMP_EQ(fs, wc, ret, X, Y, ex)				\ +  do									\ +    {									\ +      _FP_CMP_CHECK_DENORM (fs, wc, X, Y);				\ +      /* NANs are unordered.  */					\ +      if ((X##_e == _FP_EXPMAX_##fs && !_FP_FRAC_ZEROP_##wc (X))	\ +	  || (Y##_e == _FP_EXPMAX_##fs && !_FP_FRAC_ZEROP_##wc (Y)))	\ +	{								\ +	  (ret) = 1;							\ +	  _FP_CMP_CHECK_NAN (fs, wc, X, Y, (ex));			\ +	}								\ +      else								\ +	{								\ +	  _FP_CMP_CHECK_FLUSH_ZERO (fs, wc, X, Y);			\ +									\ +	  (ret) = !(X##_e == Y##_e					\ +		    && _FP_FRAC_EQ_##wc (X, Y)				\ +		    && (X##_s == Y##_s					\ +			|| (!X##_e && _FP_FRAC_ZEROP_##wc (X))));	\ +	}								\ +    }									\ +  while (0) + +/* Version to test unordered.  */ + +#define _FP_CMP_UNORD(fs, wc, ret, X, Y, ex)				\ +  do									\ +    {									\ +      _FP_CMP_CHECK_DENORM (fs, wc, X, Y);				\ +      (ret) = ((X##_e == _FP_EXPMAX_##fs && !_FP_FRAC_ZEROP_##wc (X))	\ +	       || (Y##_e == _FP_EXPMAX_##fs && !_FP_FRAC_ZEROP_##wc (Y))); \ +      if (ret)								\ +	_FP_CMP_CHECK_NAN (fs, wc, X, Y, (ex));				\ +    }									\ +  while (0) + +/* Main square root routine.  The input value should be cooked.  */ + +#define _FP_SQRT(fs, wc, R, X)						\ +  do									\ +    {									\ +      _FP_FRAC_DECL_##wc (_FP_SQRT_T);					\ +      _FP_FRAC_DECL_##wc (_FP_SQRT_S);					\ +      _FP_W_TYPE _FP_SQRT_q;						\ +      switch (X##_c)							\ +	{								\ +	case FP_CLS_NAN:						\ +	  _FP_FRAC_COPY_##wc (R, X);					\ +	  R##_s = X##_s;						\ +	  R##_c = FP_CLS_NAN;						\ +	  break;							\ +	case FP_CLS_INF:						\ +	  if (X##_s)							\ +	    {								\ +	      R##_s = _FP_NANSIGN_##fs;					\ +	      R##_c = FP_CLS_NAN; /* NAN */				\ +	      _FP_FRAC_SET_##wc (R, _FP_NANFRAC_##fs);			\ +	      FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_SQRT);	\ +	    }								\ +	  else								\ +	    {								\ +	      R##_s = 0;						\ +	      R##_c = FP_CLS_INF; /* sqrt(+inf) = +inf */		\ +	    }								\ +	  break;							\ +	case FP_CLS_ZERO:						\ +	  R##_s = X##_s;						\ +	  R##_c = FP_CLS_ZERO; /* sqrt(+-0) = +-0 */			\ +	  break;							\ +	case FP_CLS_NORMAL:						\ +	  R##_s = 0;							\ +	  if (X##_s)							\ +	    {								\ +	      R##_c = FP_CLS_NAN; /* NAN */				\ +	      R##_s = _FP_NANSIGN_##fs;					\ +	      _FP_FRAC_SET_##wc (R, _FP_NANFRAC_##fs);			\ +	      FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_SQRT);	\ +	      break;							\ +	    }								\ +	  R##_c = FP_CLS_NORMAL;					\ +	  if (X##_e & 1)						\ +	    _FP_FRAC_SLL_##wc (X, 1);					\ +	  R##_e = X##_e >> 1;						\ +	  _FP_FRAC_SET_##wc (_FP_SQRT_S, _FP_ZEROFRAC_##wc);		\ +	  _FP_FRAC_SET_##wc (R, _FP_ZEROFRAC_##wc);			\ +	  _FP_SQRT_q = _FP_OVERFLOW_##fs >> 1;				\ +	  _FP_SQRT_MEAT_##wc (R, _FP_SQRT_S, _FP_SQRT_T, X,		\ +			      _FP_SQRT_q);				\ +	}								\ +    }									\ +  while (0) + +/* Convert from FP to integer.  Input is raw.  */ + +/* RSIGNED can have following values: +   0:  the number is required to be 0..(2^rsize)-1, if not, NV is set plus +       the result is either 0 or (2^rsize)-1 depending on the sign in such +       case. +   1:  the number is required to be -(2^(rsize-1))..(2^(rsize-1))-1, if not, +       NV is set plus the result is either -(2^(rsize-1)) or (2^(rsize-1))-1 +       depending on the sign in such case. +   2:  the number is required to be -(2^(rsize-1))..(2^(rsize-1))-1, if not, +       NV is set plus the result is reduced modulo 2^rsize. +   -1: the number is required to be -(2^(rsize-1))..(2^rsize)-1, if not, NV is +       set plus the result is either -(2^(rsize-1)) or (2^(rsize-1))-1 +       depending on the sign in such case.  */ +#define _FP_TO_INT(fs, wc, r, X, rsize, rsigned)			\ +  do									\ +    {									\ +      if (X##_e < _FP_EXPBIAS_##fs)					\ +	{								\ +	  (r) = 0;							\ +	  if (X##_e == 0)						\ +	    {								\ +	      if (!_FP_FRAC_ZEROP_##wc (X))				\ +		{							\ +		  if (!FP_DENORM_ZERO)					\ +		    FP_SET_EXCEPTION (FP_EX_INEXACT);			\ +		  FP_SET_EXCEPTION (FP_EX_DENORM);			\ +		}							\ +	    }								\ +	  else								\ +	    FP_SET_EXCEPTION (FP_EX_INEXACT);				\ +	}								\ +      else if ((rsigned) == 2						\ +	       && (X##_e						\ +		   >= ((_FP_EXPMAX_##fs					\ +			< _FP_EXPBIAS_##fs + _FP_FRACBITS_##fs + (rsize) - 1) \ +		       ? _FP_EXPMAX_##fs				\ +		       : _FP_EXPBIAS_##fs + _FP_FRACBITS_##fs + (rsize) - 1))) \ +	{								\ +	  /* Overflow resulting in 0.  */				\ +	  (r) = 0;							\ +	  FP_SET_EXCEPTION (FP_EX_INVALID				\ +			    | FP_EX_INVALID_CVI				\ +			    | ((FP_EX_INVALID_SNAN			\ +				&& _FP_ISSIGNAN (fs, wc, X))		\ +			       ? FP_EX_INVALID_SNAN			\ +			       : 0));					\ +	}								\ +      else if ((rsigned) != 2						\ +	       && (X##_e >= (_FP_EXPMAX_##fs < _FP_EXPBIAS_##fs + (rsize) \ +			     ? _FP_EXPMAX_##fs				\ +			     : (_FP_EXPBIAS_##fs + (rsize)		\ +				- ((rsigned) > 0 || X##_s)))		\ +		   || (!(rsigned) && X##_s)))				\ +	{								\ +	  /* Overflow or converting to the most negative integer.  */	\ +	  if (rsigned)							\ +	    {								\ +	      (r) = 1;							\ +	      (r) <<= (rsize) - 1;					\ +	      (r) -= 1 - X##_s;						\ +	    }								\ +	  else								\ +	    {								\ +	      (r) = 0;							\ +	      if (!X##_s)						\ +		(r) = ~(r);						\ +	    }								\ +									\ +	  if (_FP_EXPBIAS_##fs + (rsize) - 1 < _FP_EXPMAX_##fs		\ +	      && (rsigned)						\ +	      && X##_s							\ +	      && X##_e == _FP_EXPBIAS_##fs + (rsize) - 1)		\ +	    {								\ +	      /* Possibly converting to most negative integer; check the \ +		 mantissa.  */						\ +	      int _FP_TO_INT_inexact = 0;				\ +	      (void) ((_FP_FRACBITS_##fs > (rsize))			\ +		      ? ({						\ +			  _FP_FRAC_SRST_##wc (X, _FP_TO_INT_inexact,	\ +					      _FP_FRACBITS_##fs - (rsize), \ +					      _FP_FRACBITS_##fs);	\ +			  0;						\ +			})						\ +		      : 0);						\ +	      if (!_FP_FRAC_ZEROP_##wc (X))				\ +		FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_CVI);	\ +	      else if (_FP_TO_INT_inexact)				\ +		FP_SET_EXCEPTION (FP_EX_INEXACT);			\ +	    }								\ +	  else								\ +	    FP_SET_EXCEPTION (FP_EX_INVALID				\ +			      | FP_EX_INVALID_CVI			\ +			      | ((FP_EX_INVALID_SNAN			\ +				  && _FP_ISSIGNAN (fs, wc, X))		\ +				 ? FP_EX_INVALID_SNAN			\ +				 : 0));					\ +	}								\ +      else								\ +	{								\ +	  int _FP_TO_INT_inexact = 0;					\ +	  _FP_FRAC_HIGH_RAW_##fs (X) |= _FP_IMPLBIT_##fs;		\ +	  if (X##_e >= _FP_EXPBIAS_##fs + _FP_FRACBITS_##fs - 1)	\ +	    {								\ +	      _FP_FRAC_ASSEMBLE_##wc ((r), X, (rsize));			\ +	      (r) <<= X##_e - _FP_EXPBIAS_##fs - _FP_FRACBITS_##fs + 1; \ +	    }								\ +	  else								\ +	    {								\ +	      _FP_FRAC_SRST_##wc (X, _FP_TO_INT_inexact,		\ +				  (_FP_FRACBITS_##fs + _FP_EXPBIAS_##fs - 1 \ +				   - X##_e),				\ +				  _FP_FRACBITS_##fs);			\ +	      _FP_FRAC_ASSEMBLE_##wc ((r), X, (rsize));			\ +	    }								\ +	  if ((rsigned) && X##_s)					\ +	    (r) = -(r);							\ +	  if ((rsigned) == 2 && X##_e >= _FP_EXPBIAS_##fs + (rsize) - 1) \ +	    {								\ +	      /* Overflow or converting to the most negative integer.  */ \ +	      if (X##_e > _FP_EXPBIAS_##fs + (rsize) - 1		\ +		  || !X##_s						\ +		  || (r) != (((__typeof__ (r)) 1) << ((rsize) - 1)))	\ +		{							\ +		  _FP_TO_INT_inexact = 0;				\ +		  FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_CVI);	\ +		}							\ +	    }								\ +	  if (_FP_TO_INT_inexact)					\ +	    FP_SET_EXCEPTION (FP_EX_INEXACT);				\ +	}								\ +    }									\ +  while (0) + +/* Convert from floating point to integer, rounding according to the +   current rounding direction.  Input is raw.  RSIGNED is as for +   _FP_TO_INT.  */ +#define _FP_TO_INT_ROUND(fs, wc, r, X, rsize, rsigned)			\ +  do									\ +    {									\ +      __label__ _FP_TO_INT_ROUND_done;					\ +      if (X##_e < _FP_EXPBIAS_##fs)					\ +	{								\ +	  int _FP_TO_INT_ROUND_rounds_away = 0;				\ +	  if (X##_e == 0)						\ +	    {								\ +	      if (_FP_FRAC_ZEROP_##wc (X))				\ +		{							\ +		  (r) = 0;						\ +		  goto _FP_TO_INT_ROUND_done;				\ +		}							\ +	      else							\ +		{							\ +		  FP_SET_EXCEPTION (FP_EX_DENORM);			\ +		  if (FP_DENORM_ZERO)					\ +		    {							\ +		      (r) = 0;						\ +		      goto _FP_TO_INT_ROUND_done;			\ +		    }							\ +		}							\ +	    }								\ +	  /* The result is 0, 1 or -1 depending on the rounding mode;	\ +	     -1 may cause overflow in the unsigned case.  */		\ +	  switch (FP_ROUNDMODE)						\ +	    {								\ +	    case FP_RND_NEAREST:					\ +	      _FP_TO_INT_ROUND_rounds_away				\ +		= (X##_e == _FP_EXPBIAS_##fs - 1			\ +		   && !_FP_FRAC_ZEROP_##wc (X));			\ +	      break;							\ +	    case FP_RND_ZERO:						\ +	      /* _FP_TO_INT_ROUND_rounds_away is already 0.  */		\ +	      break;							\ +	    case FP_RND_PINF:						\ +	      _FP_TO_INT_ROUND_rounds_away = !X##_s;			\ +	      break;							\ +	    case FP_RND_MINF:						\ +	      _FP_TO_INT_ROUND_rounds_away = X##_s;			\ +	      break;							\ +	    }								\ +	  if ((rsigned) == 0 && _FP_TO_INT_ROUND_rounds_away && X##_s)	\ +	    {								\ +	      /* Result of -1 for an unsigned conversion.  */		\ +	      (r) = 0;							\ +	      FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_CVI);	\ +	    }								\ +	  else if ((rsize) == 1 && (rsigned) > 0			\ +		   && _FP_TO_INT_ROUND_rounds_away && !X##_s)		\ +	    {								\ +	      /* Converting to a 1-bit signed bit-field, which cannot	\ +		 represent +1.  */					\ +	      (r) = ((rsigned) == 2 ? -1 : 0);				\ +	      FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_CVI);	\ +	    }								\ +	  else								\ +	    {								\ +	      (r) = (_FP_TO_INT_ROUND_rounds_away			\ +		     ? (X##_s ? -1 : 1)					\ +		     : 0);						\ +	      FP_SET_EXCEPTION (FP_EX_INEXACT);				\ +	    }								\ +	}								\ +      else if ((rsigned) == 2						\ +	       && (X##_e						\ +		   >= ((_FP_EXPMAX_##fs					\ +			< _FP_EXPBIAS_##fs + _FP_FRACBITS_##fs + (rsize) - 1) \ +		       ? _FP_EXPMAX_##fs				\ +		       : _FP_EXPBIAS_##fs + _FP_FRACBITS_##fs + (rsize) - 1))) \ +	{								\ +	  /* Overflow resulting in 0.  */				\ +	  (r) = 0;							\ +	  FP_SET_EXCEPTION (FP_EX_INVALID				\ +			    | FP_EX_INVALID_CVI				\ +			    | ((FP_EX_INVALID_SNAN			\ +				&& _FP_ISSIGNAN (fs, wc, X))		\ +			       ? FP_EX_INVALID_SNAN			\ +			       : 0));					\ +	}								\ +      else if ((rsigned) != 2						\ +	       && (X##_e >= (_FP_EXPMAX_##fs < _FP_EXPBIAS_##fs + (rsize) \ +			     ? _FP_EXPMAX_##fs				\ +			     : (_FP_EXPBIAS_##fs + (rsize)		\ +				- ((rsigned) > 0 && !X##_s)))		\ +		   || ((rsigned) == 0 && X##_s)))			\ +	{								\ +	  /* Definite overflow (does not require rounding to tell).  */	\ +	  if ((rsigned) != 0)						\ +	    {								\ +	      (r) = 1;							\ +	      (r) <<= (rsize) - 1;					\ +	      (r) -= 1 - X##_s;						\ +	    }								\ +	  else								\ +	    {								\ +	      (r) = 0;							\ +	      if (!X##_s)						\ +		(r) = ~(r);						\ +	    }								\ +									\ +	  FP_SET_EXCEPTION (FP_EX_INVALID				\ +			    | FP_EX_INVALID_CVI				\ +			    | ((FP_EX_INVALID_SNAN			\ +				&& _FP_ISSIGNAN (fs, wc, X))		\ +			       ? FP_EX_INVALID_SNAN			\ +			       : 0));					\ +	}								\ +      else								\ +	{								\ +	  /* The value is finite, with magnitude at least 1.  If	\ +	     the conversion is unsigned, the value is positive.		\ +	     If RSIGNED is not 2, the value does not definitely		\ +	     overflow by virtue of its exponent, but may still turn	\ +	     out to overflow after rounding; if RSIGNED is 2, the	\ +	     exponent may be such that the value definitely overflows,	\ +	     but at least one mantissa bit will not be shifted out.  */ \ +	  int _FP_TO_INT_ROUND_inexact = 0;				\ +	  _FP_FRAC_HIGH_RAW_##fs (X) |= _FP_IMPLBIT_##fs;		\ +	  if (X##_e >= _FP_EXPBIAS_##fs + _FP_FRACBITS_##fs - 1)	\ +	    {								\ +	      /* The value is an integer, no rounding needed.  */	\ +	      _FP_FRAC_ASSEMBLE_##wc ((r), X, (rsize));			\ +	      (r) <<= X##_e - _FP_EXPBIAS_##fs - _FP_FRACBITS_##fs + 1; \ +	    }								\ +	  else								\ +	    {								\ +	      /* May need to shift in order to round (unless there	\ +		 are exactly _FP_WORKBITS fractional bits already).  */	\ +	      int _FP_TO_INT_ROUND_rshift				\ +		= (_FP_FRACBITS_##fs + _FP_EXPBIAS_##fs			\ +		   - 1 - _FP_WORKBITS - X##_e);				\ +	      if (_FP_TO_INT_ROUND_rshift > 0)				\ +		_FP_FRAC_SRS_##wc (X, _FP_TO_INT_ROUND_rshift,		\ +				   _FP_WFRACBITS_##fs);			\ +	      else if (_FP_TO_INT_ROUND_rshift < 0)			\ +		_FP_FRAC_SLL_##wc (X, -_FP_TO_INT_ROUND_rshift);	\ +	      /* Round like _FP_ROUND, but setting			\ +		 _FP_TO_INT_ROUND_inexact instead of directly setting	\ +		 the "inexact" exception, since it may turn out we	\ +		 should set "invalid" instead.  */			\ +	      if (_FP_FRAC_LOW_##wc (X) & 7)				\ +		{							\ +		  _FP_TO_INT_ROUND_inexact = 1;				\ +		  switch (FP_ROUNDMODE)					\ +		    {							\ +		    case FP_RND_NEAREST:				\ +		      _FP_ROUND_NEAREST (wc, X);			\ +		      break;						\ +		    case FP_RND_ZERO:					\ +		      _FP_ROUND_ZERO (wc, X);				\ +		      break;						\ +		    case FP_RND_PINF:					\ +		      _FP_ROUND_PINF (wc, X);				\ +		      break;						\ +		    case FP_RND_MINF:					\ +		      _FP_ROUND_MINF (wc, X);				\ +		      break;						\ +		    }							\ +		}							\ +	      _FP_FRAC_SRL_##wc (X, _FP_WORKBITS);			\ +	      _FP_FRAC_ASSEMBLE_##wc ((r), X, (rsize));			\ +	    }								\ +	  if ((rsigned) != 0 && X##_s)					\ +	    (r) = -(r);							\ +	  /* An exponent of RSIZE - 1 always needs testing for		\ +	     overflow (either directly overflowing, or overflowing	\ +	     when rounding up results in 2^RSIZE).  An exponent of	\ +	     RSIZE - 2 can overflow for positive values when rounding	\ +	     up to 2^(RSIZE-1), but cannot overflow for negative	\ +	     values.  Smaller exponents cannot overflow.  */		\ +	  if (X##_e >= (_FP_EXPBIAS_##fs + (rsize) - 1			\ +			- ((rsigned) > 0 && !X##_s)))			\ +	    {								\ +	      if (X##_e > _FP_EXPBIAS_##fs + (rsize) - 1		\ +		  || (X##_e == _FP_EXPBIAS_##fs + (rsize) - 1		\ +		      && (X##_s						\ +			  ? (r) != (((__typeof__ (r)) 1) << ((rsize) - 1))	\ +			  : ((rsigned) > 0 || (r) == 0)))		\ +		  || ((rsigned) > 0					\ +		      && !X##_s						\ +		      && X##_e == _FP_EXPBIAS_##fs + (rsize) - 2	\ +		      && (r) == (((__typeof__ (r)) 1) << ((rsize) - 1))))	\ +		{							\ +		  if ((rsigned) != 2)					\ +		    {							\ +		      if ((rsigned) != 0)				\ +			{						\ +			  (r) = 1;					\ +			  (r) <<= (rsize) - 1;				\ +			  (r) -= 1 - X##_s;				\ +			}						\ +		      else						\ +			{						\ +			  (r) = 0;					\ +			  (r) = ~(r);					\ +			}						\ +		    }							\ +		  _FP_TO_INT_ROUND_inexact = 0;				\ +		  FP_SET_EXCEPTION (FP_EX_INVALID | FP_EX_INVALID_CVI);	\ +		}							\ +	    }								\ +	  if (_FP_TO_INT_ROUND_inexact)					\ +	    FP_SET_EXCEPTION (FP_EX_INEXACT);				\ +	}								\ +    _FP_TO_INT_ROUND_done: ;						\ +    }									\ +  while (0) + +/* Convert integer to fp.  Output is raw.  RTYPE is unsigned even if +   input is signed.  */ +#define _FP_FROM_INT(fs, wc, X, r, rsize, rtype)			\ +  do									\ +    {									\ +      __label__ pack_semiraw;						\ +      if (r)								\ +	{								\ +	  rtype _FP_FROM_INT_ur = (r);					\ +									\ +	  if ((X##_s = ((r) < 0)))					\ +	    _FP_FROM_INT_ur = -_FP_FROM_INT_ur;				\ +									\ +	  _FP_STATIC_ASSERT ((rsize) <= 2 * _FP_W_TYPE_SIZE,		\ +			     "rsize too large");			\ +	  (void) (((rsize) <= _FP_W_TYPE_SIZE)				\ +		  ? ({							\ +		      int _FP_FROM_INT_lz;				\ +		      __FP_CLZ (_FP_FROM_INT_lz,			\ +				(_FP_W_TYPE) _FP_FROM_INT_ur);		\ +		      X##_e = (_FP_EXPBIAS_##fs + _FP_W_TYPE_SIZE - 1	\ +			       - _FP_FROM_INT_lz);			\ +		    })							\ +		  : ({						\ +		      int _FP_FROM_INT_lz;				\ +		      __FP_CLZ_2 (_FP_FROM_INT_lz,			\ +				  (_FP_W_TYPE) (_FP_FROM_INT_ur		\ +						>> _FP_W_TYPE_SIZE),	\ +				  (_FP_W_TYPE) _FP_FROM_INT_ur);	\ +		      X##_e = (_FP_EXPBIAS_##fs + 2 * _FP_W_TYPE_SIZE - 1 \ +			       - _FP_FROM_INT_lz);			\ +		    }));						\ +									\ +	  if ((rsize) - 1 + _FP_EXPBIAS_##fs >= _FP_EXPMAX_##fs		\ +	      && X##_e >= _FP_EXPMAX_##fs)				\ +	    {								\ +	      /* Exponent too big; overflow to infinity.  (May also	\ +		 happen after rounding below.)  */			\ +	      _FP_OVERFLOW_SEMIRAW (fs, wc, X);				\ +	      goto pack_semiraw;					\ +	    }								\ +									\ +	  if ((rsize) <= _FP_FRACBITS_##fs				\ +	      || X##_e < _FP_EXPBIAS_##fs + _FP_FRACBITS_##fs)		\ +	    {								\ +	      /* Exactly representable; shift left.  */			\ +	      _FP_FRAC_DISASSEMBLE_##wc (X, _FP_FROM_INT_ur, (rsize));	\ +	      if (_FP_EXPBIAS_##fs + _FP_FRACBITS_##fs - 1 - X##_e > 0)	\ +		_FP_FRAC_SLL_##wc (X, (_FP_EXPBIAS_##fs			\ +				       + _FP_FRACBITS_##fs - 1 - X##_e)); \ +	    }								\ +	  else								\ +	    {								\ +	      /* More bits in integer than in floating type; need to	\ +		 round.  */						\ +	      if (_FP_EXPBIAS_##fs + _FP_WFRACBITS_##fs - 1 < X##_e)	\ +		_FP_FROM_INT_ur						\ +		  = ((_FP_FROM_INT_ur >> (X##_e - _FP_EXPBIAS_##fs	\ +					  - _FP_WFRACBITS_##fs + 1))	\ +		     | ((_FP_FROM_INT_ur				\ +			 << ((rsize) - (X##_e - _FP_EXPBIAS_##fs	\ +					- _FP_WFRACBITS_##fs + 1)))	\ +			!= 0));						\ +	      _FP_FRAC_DISASSEMBLE_##wc (X, _FP_FROM_INT_ur, (rsize));	\ +	      if ((_FP_EXPBIAS_##fs + _FP_WFRACBITS_##fs - 1 - X##_e) > 0) \ +		_FP_FRAC_SLL_##wc (X, (_FP_EXPBIAS_##fs			\ +				       + _FP_WFRACBITS_##fs - 1 - X##_e)); \ +	      _FP_FRAC_HIGH_##fs (X) &= ~(_FP_W_TYPE) _FP_IMPLBIT_SH_##fs; \ +	    pack_semiraw:						\ +	      _FP_PACK_SEMIRAW (fs, wc, X);				\ +	    }								\ +	}								\ +      else								\ +	{								\ +	  X##_s = 0;							\ +	  X##_e = 0;							\ +	  _FP_FRAC_SET_##wc (X, _FP_ZEROFRAC_##wc);			\ +	}								\ +    }									\ +  while (0) + + +/* Extend from a narrower floating-point format to a wider one.  Input +   and output are raw.  If CHECK_NAN, then signaling NaNs are +   converted to quiet with the "invalid" exception raised; otherwise +   signaling NaNs remain signaling with no exception.  */ +#define _FP_EXTEND_CNAN(dfs, sfs, dwc, swc, D, S, check_nan)		\ +  do									\ +    {									\ +      _FP_STATIC_ASSERT (_FP_FRACBITS_##dfs >= _FP_FRACBITS_##sfs,	\ +			 "destination mantissa narrower than source");	\ +      _FP_STATIC_ASSERT ((_FP_EXPMAX_##dfs - _FP_EXPBIAS_##dfs		\ +			  >= _FP_EXPMAX_##sfs - _FP_EXPBIAS_##sfs),	\ +			 "destination max exponent smaller"		\ +			 " than source");				\ +      _FP_STATIC_ASSERT (((_FP_EXPBIAS_##dfs				\ +			   >= (_FP_EXPBIAS_##sfs			\ +			       + _FP_FRACBITS_##sfs - 1))		\ +			  || (_FP_EXPBIAS_##dfs == _FP_EXPBIAS_##sfs)), \ +			 "source subnormals do not all become normal,"	\ +			 " but bias not the same");			\ +      D##_s = S##_s;							\ +      _FP_FRAC_COPY_##dwc##_##swc (D, S);				\ +      if (_FP_EXP_NORMAL (sfs, swc, S))					\ +	{								\ +	  D##_e = S##_e + _FP_EXPBIAS_##dfs - _FP_EXPBIAS_##sfs;	\ +	  _FP_FRAC_SLL_##dwc (D, (_FP_FRACBITS_##dfs - _FP_FRACBITS_##sfs)); \ +	}								\ +      else								\ +	{								\ +	  if (S##_e == 0)						\ +	    {								\ +	      _FP_CHECK_FLUSH_ZERO (sfs, swc, S);			\ +	      if (_FP_FRAC_ZEROP_##swc (S))				\ +		D##_e = 0;						\ +	      else if (_FP_EXPBIAS_##dfs				\ +		       < _FP_EXPBIAS_##sfs + _FP_FRACBITS_##sfs - 1)	\ +		{							\ +		  FP_SET_EXCEPTION (FP_EX_DENORM);			\ +		  _FP_FRAC_SLL_##dwc (D, (_FP_FRACBITS_##dfs		\ +					  - _FP_FRACBITS_##sfs));	\ +		  D##_e = 0;						\ +		  if (FP_TRAPPING_EXCEPTIONS & FP_EX_UNDERFLOW)		\ +		    FP_SET_EXCEPTION (FP_EX_UNDERFLOW);			\ +		}							\ +	      else							\ +		{							\ +		  int FP_EXTEND_lz;					\ +		  FP_SET_EXCEPTION (FP_EX_DENORM);			\ +		  _FP_FRAC_CLZ_##swc (FP_EXTEND_lz, S);			\ +		  _FP_FRAC_SLL_##dwc (D,				\ +				      FP_EXTEND_lz + _FP_FRACBITS_##dfs	\ +				      - _FP_FRACTBITS_##sfs);		\ +		  D##_e = (_FP_EXPBIAS_##dfs - _FP_EXPBIAS_##sfs + 1	\ +			   + _FP_FRACXBITS_##sfs - FP_EXTEND_lz);	\ +		}							\ +	    }								\ +	  else								\ +	    {								\ +	      D##_e = _FP_EXPMAX_##dfs;					\ +	      if (!_FP_FRAC_ZEROP_##swc (S))				\ +		{							\ +		  if (check_nan && _FP_FRAC_SNANP (sfs, S))		\ +		    FP_SET_EXCEPTION (FP_EX_INVALID			\ +				      | FP_EX_INVALID_SNAN);		\ +		  _FP_FRAC_SLL_##dwc (D, (_FP_FRACBITS_##dfs		\ +					  - _FP_FRACBITS_##sfs));	\ +		  if (check_nan)					\ +		    _FP_SETQNAN (dfs, dwc, D);				\ +		}							\ +	    }								\ +	}								\ +    }									\ +  while (0) + +#define FP_EXTEND(dfs, sfs, dwc, swc, D, S)		\ +    _FP_EXTEND_CNAN (dfs, sfs, dwc, swc, D, S, 1) + +/* Truncate from a wider floating-point format to a narrower one. +   Input and output are semi-raw.  */ +#define FP_TRUNC(dfs, sfs, dwc, swc, D, S)				\ +  do									\ +    {									\ +      _FP_STATIC_ASSERT (_FP_FRACBITS_##sfs >= _FP_FRACBITS_##dfs,	\ +			 "destination mantissa wider than source");	\ +      _FP_STATIC_ASSERT (((_FP_EXPBIAS_##sfs				\ +			   >= (_FP_EXPBIAS_##dfs			\ +			       + _FP_FRACBITS_##dfs - 1))		\ +			  || _FP_EXPBIAS_##sfs == _FP_EXPBIAS_##dfs),	\ +			 "source subnormals do not all become same,"	\ +			 " but bias not the same");			\ +      D##_s = S##_s;							\ +      if (_FP_EXP_NORMAL (sfs, swc, S))					\ +	{								\ +	  D##_e = S##_e + _FP_EXPBIAS_##dfs - _FP_EXPBIAS_##sfs;	\ +	  if (D##_e >= _FP_EXPMAX_##dfs)				\ +	    _FP_OVERFLOW_SEMIRAW (dfs, dwc, D);				\ +	  else								\ +	    {								\ +	      if (D##_e <= 0)						\ +		{							\ +		  if (D##_e < 1 - _FP_FRACBITS_##dfs)			\ +		    {							\ +		      _FP_FRAC_SET_##swc (S, _FP_ZEROFRAC_##swc);	\ +		      _FP_FRAC_LOW_##swc (S) |= 1;			\ +		    }							\ +		  else							\ +		    {							\ +		      _FP_FRAC_HIGH_##sfs (S) |= _FP_IMPLBIT_SH_##sfs;	\ +		      _FP_FRAC_SRS_##swc (S, (_FP_WFRACBITS_##sfs	\ +					      - _FP_WFRACBITS_##dfs	\ +					      + 1 - D##_e),		\ +					  _FP_WFRACBITS_##sfs);		\ +		    }							\ +		  D##_e = 0;						\ +		}							\ +	      else							\ +		_FP_FRAC_SRS_##swc (S, (_FP_WFRACBITS_##sfs		\ +					- _FP_WFRACBITS_##dfs),		\ +				    _FP_WFRACBITS_##sfs);		\ +	      _FP_FRAC_COPY_##dwc##_##swc (D, S);			\ +	    }								\ +	}								\ +      else								\ +	{								\ +	  if (S##_e == 0)						\ +	    {								\ +	      _FP_CHECK_FLUSH_ZERO (sfs, swc, S);			\ +	      D##_e = 0;						\ +	      if (_FP_FRAC_ZEROP_##swc (S))				\ +		_FP_FRAC_SET_##dwc (D, _FP_ZEROFRAC_##dwc);		\ +	      else							\ +		{							\ +		  FP_SET_EXCEPTION (FP_EX_DENORM);			\ +		  if (_FP_EXPBIAS_##sfs					\ +		      < _FP_EXPBIAS_##dfs + _FP_FRACBITS_##dfs - 1)	\ +		    {							\ +		      _FP_FRAC_SRS_##swc (S, (_FP_WFRACBITS_##sfs	\ +					      - _FP_WFRACBITS_##dfs),	\ +					  _FP_WFRACBITS_##sfs);		\ +		      _FP_FRAC_COPY_##dwc##_##swc (D, S);		\ +		    }							\ +		  else							\ +		    {							\ +		      _FP_FRAC_SET_##dwc (D, _FP_ZEROFRAC_##dwc);	\ +		      _FP_FRAC_LOW_##dwc (D) |= 1;			\ +		    }							\ +		}							\ +	    }								\ +	  else								\ +	    {								\ +	      D##_e = _FP_EXPMAX_##dfs;					\ +	      if (_FP_FRAC_ZEROP_##swc (S))				\ +		_FP_FRAC_SET_##dwc (D, _FP_ZEROFRAC_##dwc);		\ +	      else							\ +		{							\ +		  _FP_CHECK_SIGNAN_SEMIRAW (sfs, swc, S);		\ +		  _FP_FRAC_SRL_##swc (S, (_FP_WFRACBITS_##sfs		\ +					  - _FP_WFRACBITS_##dfs));	\ +		  _FP_FRAC_COPY_##dwc##_##swc (D, S);			\ +		  /* Semi-raw NaN must have all workbits cleared.  */	\ +		  _FP_FRAC_LOW_##dwc (D)				\ +		    &= ~(_FP_W_TYPE) ((1 << _FP_WORKBITS) - 1);		\ +		  _FP_SETQNAN_SEMIRAW (dfs, dwc, D);			\ +		}							\ +	    }								\ +	}								\ +    }									\ +  while (0) + +/* Helper primitives.  */ + +/* Count leading zeros in a word.  */ + +#ifndef __FP_CLZ +/* GCC 3.4 and later provide the builtins for us.  */ +# define __FP_CLZ(r, x)							\ +  do									\ +    {									\ +      _FP_STATIC_ASSERT ((sizeof (_FP_W_TYPE) == sizeof (unsigned int)	\ +			  || (sizeof (_FP_W_TYPE)			\ +			      == sizeof (unsigned long))		\ +			  || (sizeof (_FP_W_TYPE)			\ +			      == sizeof (unsigned long long))),		\ +			 "_FP_W_TYPE size unsupported for clz");	\ +      if (sizeof (_FP_W_TYPE) == sizeof (unsigned int))			\ +	(r) = __builtin_clz (x);					\ +      else if (sizeof (_FP_W_TYPE) == sizeof (unsigned long))		\ +	(r) = __builtin_clzl (x);					\ +      else /* sizeof (_FP_W_TYPE) == sizeof (unsigned long long).  */	\ +	(r) = __builtin_clzll (x);					\ +    }									\ +  while (0) +#endif /* ndef __FP_CLZ */ + +#define _FP_DIV_HELP_imm(q, r, n, d)		\ +  do						\ +    {						\ +      (q) = (n) / (d), (r) = (n) % (d);		\ +    }						\ +  while (0) + + +/* A restoring bit-by-bit division primitive.  */ + +#define _FP_DIV_MEAT_N_loop(fs, wc, R, X, Y)				\ +  do									\ +    {									\ +      int _FP_DIV_MEAT_N_loop_count = _FP_WFRACBITS_##fs;		\ +      _FP_FRAC_DECL_##wc (_FP_DIV_MEAT_N_loop_u);			\ +      _FP_FRAC_DECL_##wc (_FP_DIV_MEAT_N_loop_v);			\ +      _FP_FRAC_COPY_##wc (_FP_DIV_MEAT_N_loop_u, X);			\ +      _FP_FRAC_COPY_##wc (_FP_DIV_MEAT_N_loop_v, Y);			\ +      _FP_FRAC_SET_##wc (R, _FP_ZEROFRAC_##wc);				\ +      /* Normalize _FP_DIV_MEAT_N_LOOP_U and _FP_DIV_MEAT_N_LOOP_V.  */	\ +      _FP_FRAC_SLL_##wc (_FP_DIV_MEAT_N_loop_u, _FP_WFRACXBITS_##fs);	\ +      _FP_FRAC_SLL_##wc (_FP_DIV_MEAT_N_loop_v, _FP_WFRACXBITS_##fs);	\ +      /* First round.  Since the operands are normalized, either the	\ +	 first or second bit will be set in the fraction.  Produce a	\ +	 normalized result by checking which and adjusting the loop	\ +	 count and exponent accordingly.  */				\ +      if (_FP_FRAC_GE_1 (_FP_DIV_MEAT_N_loop_u, _FP_DIV_MEAT_N_loop_v))	\ +	{								\ +	  _FP_FRAC_SUB_##wc (_FP_DIV_MEAT_N_loop_u,			\ +			     _FP_DIV_MEAT_N_loop_u,			\ +			     _FP_DIV_MEAT_N_loop_v);			\ +	  _FP_FRAC_LOW_##wc (R) |= 1;					\ +	  _FP_DIV_MEAT_N_loop_count--;					\ +	}								\ +      else								\ +	R##_e--;							\ +      /* Subsequent rounds.  */						\ +      do								\ +	{								\ +	  int _FP_DIV_MEAT_N_loop_msb					\ +	    = (_FP_WS_TYPE) _FP_FRAC_HIGH_##wc (_FP_DIV_MEAT_N_loop_u) < 0; \ +	  _FP_FRAC_SLL_##wc (_FP_DIV_MEAT_N_loop_u, 1);			\ +	  _FP_FRAC_SLL_##wc (R, 1);					\ +	  if (_FP_DIV_MEAT_N_loop_msb					\ +	      || _FP_FRAC_GE_1 (_FP_DIV_MEAT_N_loop_u,			\ +				_FP_DIV_MEAT_N_loop_v))			\ +	    {								\ +	      _FP_FRAC_SUB_##wc (_FP_DIV_MEAT_N_loop_u,			\ +				 _FP_DIV_MEAT_N_loop_u,			\ +				 _FP_DIV_MEAT_N_loop_v);		\ +	      _FP_FRAC_LOW_##wc (R) |= 1;				\ +	    }								\ +	}								\ +      while (--_FP_DIV_MEAT_N_loop_count > 0);				\ +      /* If there's anything left in _FP_DIV_MEAT_N_LOOP_U, the result	\ +	 is inexact.  */						\ +      _FP_FRAC_LOW_##wc (R)						\ +	|= !_FP_FRAC_ZEROP_##wc (_FP_DIV_MEAT_N_loop_u);		\ +    }									\ +  while (0) + +#define _FP_DIV_MEAT_1_loop(fs, R, X, Y)  _FP_DIV_MEAT_N_loop (fs, 1, R, X, Y) +#define _FP_DIV_MEAT_2_loop(fs, R, X, Y)  _FP_DIV_MEAT_N_loop (fs, 2, R, X, Y) +#define _FP_DIV_MEAT_4_loop(fs, R, X, Y)  _FP_DIV_MEAT_N_loop (fs, 4, R, X, Y) + +#endif /* !SOFT_FP_OP_COMMON_H */ diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_add.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_add.c new file mode 100644 index 000000000..a67deab31 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_add.c @@ -0,0 +1,44 @@ +/* Software floating-point emulation. +   (*c) = (*a) + (*b) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +void _Qp_add(long double *c, const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); FP_DECL_Q(C); + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_SEMIRAW_QP(A, a); +  FP_UNPACK_SEMIRAW_QP(B, b); +  FP_ADD_Q(C, A, B); +  FP_PACK_SEMIRAW_QP(c, C); +  QP_HANDLE_EXCEPTIONS(__asm ( +"	ldd [%1], %%f52\n" +"	ldd [%1+8], %%f54\n" +"	ldd [%2], %%f56\n" +"	ldd [%2+8], %%f58\n" +"	faddq %%f52, %%f56, %%f60\n" +"	std %%f60, [%0]\n" +"	std %%f62, [%0+8]\n" +"	" : : "r" (c), "r" (a), "r" (b) : QP_CLOBBER)); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_cmp.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_cmp.c new file mode 100644 index 000000000..5316157ec --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_cmp.c @@ -0,0 +1,48 @@ +/* Software floating-point emulation. +   Compare (*a) and (*b), return float condition code. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +int _Qp_cmp(const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); +  int r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_UNPACK_RAW_QP(B, b); +  FP_CMP_Q(r, B, A, 3, 1); +  if (r == -1) r = 2; +  QP_HANDLE_EXCEPTIONS( +	__asm ( +"	ldd [%0], %%f52\n" +"	ldd [%0+8], %%f54\n" +"	ldd [%1], %%f56\n" +"	ldd [%1+8], %%f58\n" +"	fcmpq %%fcc3, %%f52, %%f56\n" +"	" : : "r" (a), "r" (b) : QP_CLOBBER_CC); +	_FPU_GETCW(_fcw); +	r = ((_fcw >> 36) & 3)); + +  return r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_cmpe.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_cmpe.c new file mode 100644 index 000000000..e0a834c72 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_cmpe.c @@ -0,0 +1,49 @@ +/* Software floating-point emulation. +   Compare (*a) and (*b), return float condition code. +   Signal exception (unless masked) if unordered. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +int _Qp_cmpe(const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); +  int r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_UNPACK_RAW_QP(B, b); +  FP_CMP_Q(r, B, A, 3, 2); +  if (r == -1) r = 2; +  QP_HANDLE_EXCEPTIONS( +	__asm ( +"	ldd [%0], %%f52\n" +"	ldd [%0+8], %%f54\n" +"	ldd [%1], %%f56\n" +"	ldd [%1+8], %%f58\n" +"	fcmpeq %%fcc3, %%f52, %%f56\n" +"	" : : "r" (a), "r" (b) : QP_CLOBBER_CC); +	_FPU_GETCW(_fcw); +	r = ((_fcw >> 36) & 3)); + +  return r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_div.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_div.c new file mode 100644 index 000000000..27d08f94d --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_div.c @@ -0,0 +1,44 @@ +/* Software floating-point emulation. +   (*c) = (*a) / (*b) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +void _Qp_div(long double *c, const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); FP_DECL_Q(C); + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_QP(A, a); +  FP_UNPACK_QP(B, b); +  FP_DIV_Q(C, A, B); +  FP_PACK_QP(c, C); +  QP_HANDLE_EXCEPTIONS(__asm ( +"	ldd [%1], %%f52\n" +"	ldd [%1+8], %%f54\n" +"	ldd [%2], %%f56\n" +"	ldd [%2+8], %%f58\n" +"	fdivq %%f52, %%f56, %%f60\n" +"	std %%f60, [%0]\n" +"	std %%f62, [%0+8]\n" +"	" : : "r" (c), "r" (a), "r" (b) : QP_CLOBBER)); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_dtoq.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_dtoq.c new file mode 100644 index 000000000..5a5c43b13 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_dtoq.c @@ -0,0 +1,45 @@ +/* Software floating-point emulation. +   (*c) = (long double)(a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "double.h" +#include "quad.h" + +void _Qp_dtoq(long double *c, const double a) +{ +  FP_DECL_EX; +  FP_DECL_D(A); +  FP_DECL_Q(C); + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_D(A, a); +#if (2 * _FP_W_TYPE_SIZE) < _FP_FRACBITS_Q +  FP_EXTEND(Q,D,4,2,C,A); +#else +  FP_EXTEND(Q,D,2,1,C,A); +#endif +  FP_PACK_RAW_QP(c, C); +  QP_HANDLE_EXCEPTIONS(__asm ( +"	fdtoq %1, %%f60\n" +"	std %%f60, [%0]\n" +"	std %%f62, [%0+8]\n" +"	" : : "r" (c), "e" (a) : QP_CLOBBER)); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_feq.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_feq.c new file mode 100644 index 000000000..c7c626378 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_feq.c @@ -0,0 +1,48 @@ +/* Software floating-point emulation. +   Return 1 if (*a) == (*b) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +int _Qp_feq(const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); +  int r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_UNPACK_RAW_QP(B, b); +  FP_CMP_EQ_Q(r, A, B, 1); + +  QP_HANDLE_EXCEPTIONS( +	__asm ( +"	ldd [%0], %%f52\n" +"	ldd [%0+8], %%f54\n" +"	ldd [%1], %%f56\n" +"	ldd [%1+8], %%f58\n" +"	fcmpq %%fcc3, %%f52, %%f56\n" +"	" : : "r" (a), "r" (b) : QP_CLOBBER_CC); +	_FPU_GETCW(_fcw); +	r = ((_fcw >> 36) & 3)); + +  return !r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_fge.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_fge.c new file mode 100644 index 000000000..19cacbb34 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_fge.c @@ -0,0 +1,48 @@ +/* Software floating-point emulation. +   Return 1 if (*a) >= (*b) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +int _Qp_fge(const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); +  int r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_UNPACK_RAW_QP(B, b); +  FP_CMP_Q(r, B, A, 3, 2); + +  QP_HANDLE_EXCEPTIONS( +	__asm ( +"	ldd [%0], %%f52\n" +"	ldd [%0+8], %%f54\n" +"	ldd [%1], %%f56\n" +"	ldd [%1+8], %%f58\n" +"	fcmpeq %%fcc3, %%f52, %%f56\n" +"	" : : "r" (a), "r" (b) : QP_CLOBBER_CC); +	_FPU_GETCW(_fcw); +	r = ((_fcw >> 36) & 1)); + +  return (r <= 0); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_fgt.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_fgt.c new file mode 100644 index 000000000..70645d1cb --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_fgt.c @@ -0,0 +1,48 @@ +/* Software floating-point emulation. +   Return 1 if (*a) > (*b) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +int _Qp_fgt(const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); +  int r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_UNPACK_RAW_QP(B, b); +  FP_CMP_Q(r, B, A, 3, 2); + +  QP_HANDLE_EXCEPTIONS( +	__asm ( +"	ldd [%0], %%f52\n" +"	ldd [%0+8], %%f54\n" +"	ldd [%1], %%f56\n" +"	ldd [%1+8], %%f58\n" +"	fcmpeq %%fcc3, %%f52, %%f56\n" +"	" : : "r" (a), "r" (b) : QP_CLOBBER_CC); +	_FPU_GETCW(_fcw); +	r = ((_fcw >> 36) & 3) - 3); + +  return (r == -1); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_fle.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_fle.c new file mode 100644 index 000000000..6293fcbcd --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_fle.c @@ -0,0 +1,48 @@ +/* Software floating-point emulation. +   Return 1 if (*a) <= (*b) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +int _Qp_fle(const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); +  int r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_UNPACK_RAW_QP(B, b); +  FP_CMP_Q(r, B, A, -2, 2); + +  QP_HANDLE_EXCEPTIONS( +	__asm ( +"	ldd [%0], %%f52\n" +"	ldd [%0+8], %%f54\n" +"	ldd [%1], %%f56\n" +"	ldd [%1+8], %%f58\n" +"	fcmpeq %%fcc3, %%f52, %%f56\n" +"	" : : "r" (a), "r" (b) : QP_CLOBBER_CC); +	_FPU_GETCW(_fcw); +	r = ((_fcw >> 36) & 2) ? -1 : 0); + +  return (r >= 0); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_flt.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_flt.c new file mode 100644 index 000000000..7aa054697 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_flt.c @@ -0,0 +1,48 @@ +/* Software floating-point emulation. +   Return 1 if (*a) < (*b) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +int _Qp_flt(const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); +  int r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_UNPACK_RAW_QP(B, b); +  FP_CMP_Q(r, B, A, 3, 2); + +  QP_HANDLE_EXCEPTIONS( +	__asm ( +"	ldd [%0], %%f52\n" +"	ldd [%0+8], %%f54\n" +"	ldd [%1], %%f56\n" +"	ldd [%1+8], %%f58\n" +"	fcmpeq %%fcc3, %%f52, %%f56\n" +"	" : : "r" (a), "r" (b) : QP_CLOBBER_CC); +	_FPU_GETCW(_fcw); +	r = ((_fcw >> 36) & 3)); + +  return (r == 1); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_fne.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_fne.c new file mode 100644 index 000000000..dd358eda0 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_fne.c @@ -0,0 +1,49 @@ +/* Software floating-point emulation. +   Return 1 if (*a) != (*b) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +int _Qp_fne(const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); +  int r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_UNPACK_RAW_QP(B, b); +  FP_CMP_EQ_Q(r, A, B, 1); + +  QP_HANDLE_EXCEPTIONS( +	__asm ( +"	ldd [%0], %%f52\n" +"	ldd [%0+8], %%f54\n" +"	ldd [%1], %%f56\n" +"	ldd [%1+8], %%f58\n" +"	fcmpq %%fcc3, %%f52, %%f56\n" +"	" : : "r" (a), "r" (b) : QP_CLOBBER_CC); +	_FPU_GETCW(_fcw); +	r = ((_fcw >> 36) & 3) != 0); + + +  return r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_itoq.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_itoq.c new file mode 100644 index 000000000..230fde365 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_itoq.c @@ -0,0 +1,34 @@ +/* Software floating-point emulation. +   (*c) = (long double)(a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +void _Qp_itoq(long double *c, const int a) +{ +  FP_DECL_EX; +  FP_DECL_Q(C); +  int b = a; + +  FP_FROM_INT_Q(C, b, 32, unsigned int); +  FP_PACK_RAW_QP(c, C); +  QP_NO_EXCEPTIONS; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_mul.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_mul.c new file mode 100644 index 000000000..49a290af9 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_mul.c @@ -0,0 +1,49 @@ +/* Software floating-point emulation. +   (*c) = (*a) * (*b) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +/* As QP_HANDLE_EXCEPTIONS reloads FPU control word anyway, +   avoid doing it twice.  */ +#define _FP_MUL_MEAT_RESET_FE do {} while (0) +#include "soft-fp.h" +#include "quad.h" + +void _Qp_mul(long double *c, const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); FP_DECL_Q(C); + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_QP(A, a); +  FP_UNPACK_QP(B, b); +  FP_MUL_Q(C, A, B); +  FP_PACK_QP(c, C); +  QP_HANDLE_EXCEPTIONS( +	_FPU_SETCW(_fcw); +	__asm ( +"	ldd [%1], %%f52\n" +"	ldd [%1+8], %%f54\n" +"	ldd [%2], %%f56\n" +"	ldd [%2+8], %%f58\n" +"	fmulq %%f52, %%f56, %%f60\n" +"	std %%f60, [%0]\n" +"	std %%f62, [%0+8]\n" +"	" : : "r" (c), "r" (a), "r" (b) : QP_CLOBBER)); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_neg.S b/libc/sysdeps/linux/sparc64/soft-fp/qp_neg.S new file mode 100644 index 000000000..d2fd7f286 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_neg.S @@ -0,0 +1,30 @@ +/* Quad floating-point emulation. +   (*c) = !(*a) +   Copyright (C) 1999-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <sysdep.h> + +ENTRY(_Qp_neg) +	ldd	[%o1], %f60 +	ldd	[%o1 + 8], %f62 +	fnegd	%f60, %f60 +	std	%f60, [%o0] +	jmpl	%o7 + 8, %g0 +	 std	%f62, [%o0 + 8] +END(_Qp_neg) diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_qtod.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtod.c new file mode 100644 index 000000000..7c3889da9 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtod.c @@ -0,0 +1,48 @@ +/* Software floating-point emulation. +   Return (double)(*a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "double.h" +#include "quad.h" + +double _Qp_qtod(const long double *a) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); +  FP_DECL_D(R); +  double r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_SEMIRAW_QP(A, a); +#if (2 * _FP_W_TYPE_SIZE) < _FP_FRACBITS_Q +  FP_TRUNC(D,Q,2,4,R,A); +#else +  FP_TRUNC(D,Q,1,2,R,A); +#endif +  FP_PACK_SEMIRAW_D(r, R); +  QP_HANDLE_EXCEPTIONS(__asm ( +"	ldd [%1], %%f52\n" +"	ldd [%1+8], %%f54\n" +"	fqtod %%f52, %0\n" +"	" : "=&e" (r) : "r" (a) : QP_CLOBBER)); + +  return r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_qtoi.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtoi.c new file mode 100644 index 000000000..99cd760ac --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtoi.c @@ -0,0 +1,46 @@ +/* Software floating-point emulation. +   Return (int)(*a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#define FP_ROUNDMODE FP_RND_ZERO +#include "soft-fp.h" +#include "quad.h" + +int _Qp_qtoi(const long double *a) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); +  unsigned int r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_TO_INT_Q(r, A, 32, 1); +  QP_HANDLE_EXCEPTIONS( +	int rx; +	__asm ( +"	ldd [%1], %%f52\n" +"	ldd [%1+8], %%f54\n" +"	fqtoi %%f52, %%f31\n" +"	st %%f31, [%0]\n" +"	" : : "r" (&rx), "r" (a) : QP_CLOBBER, "f31"); +	r = rx); + +  return r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_qtos.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtos.c new file mode 100644 index 000000000..dacd6c062 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtos.c @@ -0,0 +1,49 @@ +/* Software floating-point emulation. +   Return (float)(*a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "single.h" +#include "quad.h" + +float _Qp_qtos(const long double *a) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); +  FP_DECL_S(R); +  float r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_SEMIRAW_QP(A, a); +#if (2 * _FP_W_TYPE_SIZE) < _FP_FRACBITS_Q +  FP_TRUNC(S,Q,1,4,R,A); +#else +  FP_TRUNC(S,Q,1,2,R,A); +#endif +  FP_PACK_SEMIRAW_S(r, R); + +  QP_HANDLE_EXCEPTIONS(__asm ( +"	ldd [%1], %%f52\n" +"	ldd [%1+8], %%f54\n" +"	fqtos %%f52, %0\n" +"	" : "=&f" (r) : "r" (a) : QP_CLOBBER)); + +  return r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_qtoui.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtoui.c new file mode 100644 index 000000000..2d8fb5253 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtoui.c @@ -0,0 +1,46 @@ +/* Software floating-point emulation. +   Return (unsigned int)(*a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#define FP_ROUNDMODE FP_RND_ZERO +#include "soft-fp.h" +#include "quad.h" + +unsigned int _Qp_qtoui(const long double *a) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); +  unsigned int r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_TO_INT_Q(r, A, 32, -1); +  QP_HANDLE_EXCEPTIONS( +	int rx; +	__asm ( +"	ldd [%1], %%f52\n" +"	ldd [%1+8], %%f54\n" +"	fqtoi %%f52, %%f31\n" +"	st %%f31, [%0]\n" +"	" : : "r" (&rx), "r" (a) : QP_CLOBBER, "f31"); +	r = rx); + +  return r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_qtoux.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtoux.c new file mode 100644 index 000000000..2d74a6ee1 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtoux.c @@ -0,0 +1,46 @@ +/* Software floating-point emulation. +   Return (unsigned long)(*a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#define FP_ROUNDMODE FP_RND_ZERO +#include "soft-fp.h" +#include "quad.h" + +unsigned long _Qp_qtoux(const long double *a) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); +  unsigned long r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_TO_INT_Q(r, A, 64, -1); +  QP_HANDLE_EXCEPTIONS( +	unsigned long rx; +	__asm ( +"	ldd [%1], %%f52\n" +"	ldd [%1+8], %%f54\n" +"	fqtox %%f52, %%f60\n" +"	std %%f60, [%0]\n" +"	" : : "r" (&rx), "r" (a) : QP_CLOBBER); +	r = rx); + +  return r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_qtox.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtox.c new file mode 100644 index 000000000..abfc666cd --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_qtox.c @@ -0,0 +1,46 @@ +/* Software floating-point emulation. +   Return (long)(*a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#define FP_ROUNDMODE FP_RND_ZERO +#include "soft-fp.h" +#include "quad.h" + +long _Qp_qtox(const long double *a) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); +  unsigned long r; + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_QP(A, a); +  FP_TO_INT_Q(r, A, 64, 1); +  QP_HANDLE_EXCEPTIONS( +	long rx; +	__asm ( +"	ldd [%1], %%f52\n" +"	ldd [%1+8], %%f54\n" +"	fqtox %%f52, %%f60\n" +"	std %%f60, [%0]\n" +"	" : : "r" (&rx), "r" (a) : QP_CLOBBER); +	r = rx); + +  return r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_sqrt.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_sqrt.c new file mode 100644 index 000000000..3d78b1fdb --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_sqrt.c @@ -0,0 +1,41 @@ +/* Software floating-point emulation. +   (*c) = sqrtl(*a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +void _Qp_sqrt(long double *c, const long double *a) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(C); + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_QP(A, a); +  FP_SQRT_Q(C, A); +  FP_PACK_QP(c, C); +  QP_HANDLE_EXCEPTIONS(__asm ( +"	ldd [%1], %%f52\n" +"	ldd [%1+8], %%f54\n" +"	fsqrtq %%f52, %%f60\n" +"	std %%f60, [%0]\n" +"	std %%f62, [%0+8]\n" +"	" : : "r" (c), "r" (a) : QP_CLOBBER)); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_stoq.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_stoq.c new file mode 100644 index 000000000..9202a7269 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_stoq.c @@ -0,0 +1,45 @@ +/* Software floating-point emulation. +   (*c) = (long double)(a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "single.h" +#include "quad.h" + +void _Qp_stoq(long double *c, const float a) +{ +  FP_DECL_EX; +  FP_DECL_S(A); +  FP_DECL_Q(C); + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_RAW_S(A, a); +#if (2 * _FP_W_TYPE_SIZE) < _FP_FRACBITS_Q +  FP_EXTEND(Q,S,4,1,C,A); +#else +  FP_EXTEND(Q,S,2,1,C,A); +#endif +  FP_PACK_RAW_QP(c, C); +  QP_HANDLE_EXCEPTIONS(__asm ( +"	fstoq %1, %%f60\n" +"	std %%f60, [%0]\n" +"	std %%f62, [%0+8]\n" +"	" : : "r" (c), "f" (a) : QP_CLOBBER)); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_sub.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_sub.c new file mode 100644 index 000000000..71b989074 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_sub.c @@ -0,0 +1,44 @@ +/* Software floating-point emulation. +   (*c) = (*a) - (*b) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +void _Qp_sub(long double *c, const long double *a, const long double *b) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); FP_DECL_Q(B); FP_DECL_Q(C); + +  FP_INIT_ROUNDMODE; +  FP_UNPACK_SEMIRAW_QP(A, a); +  FP_UNPACK_SEMIRAW_QP(B, b); +  FP_SUB_Q(C, A, B); +  FP_PACK_SEMIRAW_QP(c, C); +  QP_HANDLE_EXCEPTIONS(__asm ( +"	ldd [%1], %%f52\n" +"	ldd [%1+8], %%f54\n" +"	ldd [%2], %%f56\n" +"	ldd [%2+8], %%f58\n" +"	fsubq %%f52, %%f56, %%f60\n" +"	std %%f60, [%0]\n" +"	std %%f62, [%0+8]\n" +"	" : : "r" (c), "r" (a), "r" (b) : QP_CLOBBER)); +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_uitoq.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_uitoq.c new file mode 100644 index 000000000..cc8603ad4 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_uitoq.c @@ -0,0 +1,34 @@ +/* Software floating-point emulation. +   (*c) = (long double)(a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +void _Qp_uitoq(long double *c, const unsigned int a) +{ +  FP_DECL_EX; +  FP_DECL_Q(C); +  unsigned int b = a; + +  FP_FROM_INT_Q(C, b, 32, unsigned int); +  FP_PACK_RAW_QP(c, C); +  QP_NO_EXCEPTIONS; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_util.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_util.c new file mode 100644 index 000000000..7e1f2511e --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_util.c @@ -0,0 +1,60 @@ +/* Software floating-point emulation. +   Helper routine for _Qp_* routines. +   Simulate exceptions using double arithmetics. +   Copyright (C) 1999-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <float.h> +#include <math.h> +#include <assert.h> +#include "soft-fp.h" + +void __Qp_handle_exceptions(int exceptions) +{ +  if (exceptions & FP_EX_INVALID) +    { +      float f = 0.0; +      __asm__ __volatile__ ("fdivs %0, %0, %0" : "+f" (f)); +    } +  if (exceptions & FP_EX_DIVZERO) +    { +      float f = 1.0, g = 0.0; +      __asm__ __volatile__ ("fdivs %0, %1, %0" +			    : "+f" (f) +			    : "f" (g)); +    } +  if (exceptions & FP_EX_OVERFLOW) +    { +      float f = FLT_MAX; +      __asm__ __volatile__("fmuls %0, %0, %0" : "+f" (f)); +      exceptions &= ~FP_EX_INEXACT; +    } +  if (exceptions & FP_EX_UNDERFLOW) +    { +      float f = FLT_MIN; +      __asm__ __volatile__("fmuls %0, %0, %0" : "+f" (f)); +      exceptions &= ~FP_EX_INEXACT; +    } +  if (exceptions & FP_EX_INEXACT) +    { +      double d = 1.0, e = M_PI; +      __asm__ __volatile__ ("fdivd %0, %1, %0" +			    : "+f" (d) +			    : "f" (e)); +    } +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_uxtoq.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_uxtoq.c new file mode 100644 index 000000000..766ca7887 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_uxtoq.c @@ -0,0 +1,34 @@ +/* Software floating-point emulation. +   (*c) = (long double)(a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +void _Qp_uxtoq(long double *c, const unsigned long a) +{ +  FP_DECL_EX; +  FP_DECL_Q(C); +  unsigned long b = a; + +  FP_FROM_INT_Q(C, b, 64, unsigned long); +  FP_PACK_RAW_QP(c, C); +  QP_NO_EXCEPTIONS; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/qp_xtoq.c b/libc/sysdeps/linux/sparc64/soft-fp/qp_xtoq.c new file mode 100644 index 000000000..42c0bf104 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/qp_xtoq.c @@ -0,0 +1,34 @@ +/* Software floating-point emulation. +   (*c) = (long double)(*a) +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com) and +		  Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include "soft-fp.h" +#include "quad.h" + +void _Qp_xtoq(long double *c, const long a) +{ +  FP_DECL_EX; +  FP_DECL_Q(C); +  long b = a; + +  FP_FROM_INT_Q(C, b, 64, unsigned long); +  FP_PACK_RAW_QP(c, C); +  QP_NO_EXCEPTIONS; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/quad.h b/libc/sysdeps/linux/sparc64/soft-fp/quad.h new file mode 100644 index 000000000..388601fd8 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/quad.h @@ -0,0 +1,330 @@ +/* Software floating-point emulation. +   Definitions for IEEE Quad Precision. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com), +		  Jakub Jelinek (jj@ultra.linux.cz), +		  David S. Miller (davem@redhat.com) and +		  Peter Maydell (pmaydell@chiark.greenend.org.uk). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 of the License, or (at your option) any later version. + +   In addition to the permissions in the GNU Lesser General Public +   License, the Free Software Foundation gives you unlimited +   permission to link the compiled version of this file into +   combinations with other programs, and to distribute those +   combinations without any restriction coming from the use of this +   file.  (The Lesser General Public License restrictions do apply in +   other respects; for example, they cover modification of the file, +   and distribution when not linked into a combine executable.) + +   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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef SOFT_FP_QUAD_H +#define SOFT_FP_QUAD_H	1 + +#if _FP_W_TYPE_SIZE < 32 +# error "Here's a nickel, kid. Go buy yourself a real computer." +#endif + +#if _FP_W_TYPE_SIZE < 64 +# define _FP_FRACTBITS_Q	(4*_FP_W_TYPE_SIZE) +# define _FP_FRACTBITS_DW_Q	(8*_FP_W_TYPE_SIZE) +#else +# define _FP_FRACTBITS_Q		(2*_FP_W_TYPE_SIZE) +# define _FP_FRACTBITS_DW_Q	(4*_FP_W_TYPE_SIZE) +#endif + +#define _FP_FRACBITS_Q		113 +#define _FP_FRACXBITS_Q		(_FP_FRACTBITS_Q - _FP_FRACBITS_Q) +#define _FP_WFRACBITS_Q		(_FP_WORKBITS + _FP_FRACBITS_Q) +#define _FP_WFRACXBITS_Q	(_FP_FRACTBITS_Q - _FP_WFRACBITS_Q) +#define _FP_EXPBITS_Q		15 +#define _FP_EXPBIAS_Q		16383 +#define _FP_EXPMAX_Q		32767 + +#define _FP_QNANBIT_Q		\ +	((_FP_W_TYPE) 1 << (_FP_FRACBITS_Q-2) % _FP_W_TYPE_SIZE) +#define _FP_QNANBIT_SH_Q		\ +	((_FP_W_TYPE) 1 << (_FP_FRACBITS_Q-2+_FP_WORKBITS) % _FP_W_TYPE_SIZE) +#define _FP_IMPLBIT_Q		\ +	((_FP_W_TYPE) 1 << (_FP_FRACBITS_Q-1) % _FP_W_TYPE_SIZE) +#define _FP_IMPLBIT_SH_Q		\ +	((_FP_W_TYPE) 1 << (_FP_FRACBITS_Q-1+_FP_WORKBITS) % _FP_W_TYPE_SIZE) +#define _FP_OVERFLOW_Q		\ +	((_FP_W_TYPE) 1 << (_FP_WFRACBITS_Q % _FP_W_TYPE_SIZE)) + +#define _FP_WFRACBITS_DW_Q	(2 * _FP_WFRACBITS_Q) +#define _FP_WFRACXBITS_DW_Q	(_FP_FRACTBITS_DW_Q - _FP_WFRACBITS_DW_Q) +#define _FP_HIGHBIT_DW_Q	\ +  ((_FP_W_TYPE) 1 << (_FP_WFRACBITS_DW_Q - 1) % _FP_W_TYPE_SIZE) + +typedef float TFtype __attribute__ ((mode (TF))); + +#if _FP_W_TYPE_SIZE < 64 + +union _FP_UNION_Q +{ +  TFtype flt; +  struct _FP_STRUCT_LAYOUT +  { +# if __BYTE_ORDER == __BIG_ENDIAN +    unsigned sign : 1; +    unsigned exp : _FP_EXPBITS_Q; +    unsigned long frac3 : _FP_FRACBITS_Q - (_FP_IMPLBIT_Q != 0)-(_FP_W_TYPE_SIZE * 3); +    unsigned long frac2 : _FP_W_TYPE_SIZE; +    unsigned long frac1 : _FP_W_TYPE_SIZE; +    unsigned long frac0 : _FP_W_TYPE_SIZE; +# else +    unsigned long frac0 : _FP_W_TYPE_SIZE; +    unsigned long frac1 : _FP_W_TYPE_SIZE; +    unsigned long frac2 : _FP_W_TYPE_SIZE; +    unsigned long frac3 : _FP_FRACBITS_Q - (_FP_IMPLBIT_Q != 0)-(_FP_W_TYPE_SIZE * 3); +    unsigned exp : _FP_EXPBITS_Q; +    unsigned sign : 1; +# endif /* not bigendian */ +  } bits __attribute__ ((packed)); +}; + + +# define FP_DECL_Q(X)		_FP_DECL (4, X) +# define FP_UNPACK_RAW_Q(X, val)	_FP_UNPACK_RAW_4 (Q, X, (val)) +# define FP_UNPACK_RAW_QP(X, val)	_FP_UNPACK_RAW_4_P (Q, X, (val)) +# define FP_PACK_RAW_Q(val, X)	_FP_PACK_RAW_4 (Q, (val), X) +# define FP_PACK_RAW_QP(val, X)			\ +  do						\ +    {						\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_4_P (Q, (val), X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_Q(X, val)			\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_4 (Q, X, (val));		\ +      _FP_UNPACK_CANONICAL (Q, 4, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_QP(X, val)			\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_4_P (Q, X, (val));		\ +      _FP_UNPACK_CANONICAL (Q, 4, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_SEMIRAW_Q(X, val)		\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_4 (Q, X, (val));		\ +      _FP_UNPACK_SEMIRAW (Q, 4, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_SEMIRAW_QP(X, val)		\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_4_P (Q, X, (val));		\ +      _FP_UNPACK_SEMIRAW (Q, 4, X);		\ +    }						\ +  while (0) + +# define FP_PACK_Q(val, X)			\ +  do						\ +    {						\ +      _FP_PACK_CANONICAL (Q, 4, X);		\ +      _FP_PACK_RAW_4 (Q, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_QP(val, X)			\ +  do						\ +    {						\ +      _FP_PACK_CANONICAL (Q, 4, X);		\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_4_P (Q, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_SEMIRAW_Q(val, X)		\ +  do						\ +    {						\ +      _FP_PACK_SEMIRAW (Q, 4, X);		\ +      _FP_PACK_RAW_4 (Q, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_SEMIRAW_QP(val, X)		\ +  do						\ +    {						\ +      _FP_PACK_SEMIRAW (Q, 4, X);		\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_4_P (Q, (val), X);		\ +    }						\ +  while (0) + +# define FP_ISSIGNAN_Q(X)		_FP_ISSIGNAN (Q, 4, X) +# define FP_NEG_Q(R, X)			_FP_NEG (Q, 4, R, X) +# define FP_ADD_Q(R, X, Y)		_FP_ADD (Q, 4, R, X, Y) +# define FP_SUB_Q(R, X, Y)		_FP_SUB (Q, 4, R, X, Y) +# define FP_MUL_Q(R, X, Y)		_FP_MUL (Q, 4, R, X, Y) +# define FP_DIV_Q(R, X, Y)		_FP_DIV (Q, 4, R, X, Y) +# define FP_SQRT_Q(R, X)		_FP_SQRT (Q, 4, R, X) +# define _FP_SQRT_MEAT_Q(R, S, T, X, Q)	_FP_SQRT_MEAT_4 (R, S, T, X, (Q)) +# define FP_FMA_Q(R, X, Y, Z)		_FP_FMA (Q, 4, 8, R, X, Y, Z) + +# define FP_CMP_Q(r, X, Y, un, ex)	_FP_CMP (Q, 4, (r), X, Y, (un), (ex)) +# define FP_CMP_EQ_Q(r, X, Y, ex)	_FP_CMP_EQ (Q, 4, (r), X, Y, (ex)) +# define FP_CMP_UNORD_Q(r, X, Y, ex)	_FP_CMP_UNORD (Q, 4, (r), X, Y, (ex)) + +# define FP_TO_INT_Q(r, X, rsz, rsg)	_FP_TO_INT (Q, 4, (r), X, (rsz), (rsg)) +# define FP_TO_INT_ROUND_Q(r, X, rsz, rsg)	\ +  _FP_TO_INT_ROUND (Q, 4, (r), X, (rsz), (rsg)) +# define FP_FROM_INT_Q(X, r, rs, rt)	_FP_FROM_INT (Q, 4, X, (r), (rs), rt) + +# define _FP_FRAC_HIGH_Q(X)	_FP_FRAC_HIGH_4 (X) +# define _FP_FRAC_HIGH_RAW_Q(X)	_FP_FRAC_HIGH_4 (X) + +# define _FP_FRAC_HIGH_DW_Q(X)	_FP_FRAC_HIGH_8 (X) + +#else   /* not _FP_W_TYPE_SIZE < 64 */ +union _FP_UNION_Q +{ +  TFtype flt /* __attribute__ ((mode (TF))) */ ; +  struct _FP_STRUCT_LAYOUT +  { +    _FP_W_TYPE a, b; +  } longs; +  struct _FP_STRUCT_LAYOUT +  { +# if __BYTE_ORDER == __BIG_ENDIAN +    unsigned sign    : 1; +    unsigned exp     : _FP_EXPBITS_Q; +    _FP_W_TYPE frac1 : _FP_FRACBITS_Q - (_FP_IMPLBIT_Q != 0) - _FP_W_TYPE_SIZE; +    _FP_W_TYPE frac0 : _FP_W_TYPE_SIZE; +# else +    _FP_W_TYPE frac0 : _FP_W_TYPE_SIZE; +    _FP_W_TYPE frac1 : _FP_FRACBITS_Q - (_FP_IMPLBIT_Q != 0) - _FP_W_TYPE_SIZE; +    unsigned exp     : _FP_EXPBITS_Q; +    unsigned sign    : 1; +# endif +  } bits; +}; + +# define FP_DECL_Q(X)		_FP_DECL (2, X) +# define FP_UNPACK_RAW_Q(X, val)	_FP_UNPACK_RAW_2 (Q, X, (val)) +# define FP_UNPACK_RAW_QP(X, val)	_FP_UNPACK_RAW_2_P (Q, X, (val)) +# define FP_PACK_RAW_Q(val, X)	_FP_PACK_RAW_2 (Q, (val), X) +# define FP_PACK_RAW_QP(val, X)			\ +  do						\ +    {						\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_2_P (Q, (val), X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_Q(X, val)			\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_2 (Q, X, (val));		\ +      _FP_UNPACK_CANONICAL (Q, 2, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_QP(X, val)			\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_2_P (Q, X, (val));		\ +      _FP_UNPACK_CANONICAL (Q, 2, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_SEMIRAW_Q(X, val)		\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_2 (Q, X, (val));		\ +      _FP_UNPACK_SEMIRAW (Q, 2, X);		\ +    }						\ +  while (0) + +# define FP_UNPACK_SEMIRAW_QP(X, val)		\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_2_P (Q, X, (val));		\ +      _FP_UNPACK_SEMIRAW (Q, 2, X);		\ +    }						\ +  while (0) + +# define FP_PACK_Q(val, X)			\ +  do						\ +    {						\ +      _FP_PACK_CANONICAL (Q, 2, X);		\ +      _FP_PACK_RAW_2 (Q, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_QP(val, X)			\ +  do						\ +    {						\ +      _FP_PACK_CANONICAL (Q, 2, X);		\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_2_P (Q, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_SEMIRAW_Q(val, X)		\ +  do						\ +    {						\ +      _FP_PACK_SEMIRAW (Q, 2, X);		\ +      _FP_PACK_RAW_2 (Q, (val), X);		\ +    }						\ +  while (0) + +# define FP_PACK_SEMIRAW_QP(val, X)		\ +  do						\ +    {						\ +      _FP_PACK_SEMIRAW (Q, 2, X);		\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_2_P (Q, (val), X);		\ +    }						\ +  while (0) + +# define FP_ISSIGNAN_Q(X)		_FP_ISSIGNAN (Q, 2, X) +# define FP_NEG_Q(R, X)			_FP_NEG (Q, 2, R, X) +# define FP_ADD_Q(R, X, Y)		_FP_ADD (Q, 2, R, X, Y) +# define FP_SUB_Q(R, X, Y)		_FP_SUB (Q, 2, R, X, Y) +# define FP_MUL_Q(R, X, Y)		_FP_MUL (Q, 2, R, X, Y) +# define FP_DIV_Q(R, X, Y)		_FP_DIV (Q, 2, R, X, Y) +# define FP_SQRT_Q(R, X)		_FP_SQRT (Q, 2, R, X) +# define _FP_SQRT_MEAT_Q(R, S, T, X, Q)	_FP_SQRT_MEAT_2 (R, S, T, X, (Q)) +# define FP_FMA_Q(R, X, Y, Z)		_FP_FMA (Q, 2, 4, R, X, Y, Z) + +# define FP_CMP_Q(r, X, Y, un, ex)	_FP_CMP (Q, 2, (r), X, Y, (un), (ex)) +# define FP_CMP_EQ_Q(r, X, Y, ex)	_FP_CMP_EQ (Q, 2, (r), X, Y, (ex)) +# define FP_CMP_UNORD_Q(r, X, Y, ex)	_FP_CMP_UNORD (Q, 2, (r), X, Y, (ex)) + +# define FP_TO_INT_Q(r, X, rsz, rsg)	_FP_TO_INT (Q, 2, (r), X, (rsz), (rsg)) +# define FP_TO_INT_ROUND_Q(r, X, rsz, rsg)	\ +  _FP_TO_INT_ROUND (Q, 2, (r), X, (rsz), (rsg)) +# define FP_FROM_INT_Q(X, r, rs, rt)	_FP_FROM_INT (Q, 2, X, (r), (rs), rt) + +# define _FP_FRAC_HIGH_Q(X)	_FP_FRAC_HIGH_2 (X) +# define _FP_FRAC_HIGH_RAW_Q(X)	_FP_FRAC_HIGH_2 (X) + +# define _FP_FRAC_HIGH_DW_Q(X)	_FP_FRAC_HIGH_4 (X) + +#endif /* not _FP_W_TYPE_SIZE < 64 */ + +#endif /* !SOFT_FP_QUAD_H */ diff --git a/libc/sysdeps/linux/sparc64/soft-fp/s_frexpl.c b/libc/sysdeps/linux/sparc64/soft-fp/s_frexpl.c new file mode 100644 index 000000000..6f0baeb3f --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/s_frexpl.c @@ -0,0 +1,51 @@ +/* Software floating-point emulation. +   frexpl(x, exp) +   Copyright (C) 1999-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +/* + * for non-zero x + *      x = frexpl(arg,&exp); + * return a long double fp quantity x such that 0.5 <= |x| <1.0 + * and the corresponding binary exponent "exp". That is + *      arg = x*2^exp. + * If arg is inf, 0.0, or NaN, then frexpl(arg,&exp) returns arg + * with *exp=0. + */ + +#include "soft-fp.h" +#include "quad.h" + +long double __frexpl(long double arg, int *exp) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); +  long double r; + +  *exp = 0; +  FP_UNPACK_Q(A, arg); +  if (A_c != FP_CLS_NORMAL) +    return arg; +  *exp = A_e + 1; +  A_e = -1; +  FP_PACK_Q(r, A); + +  return r; +} + +weak_alias (__frexpl, frexpl) diff --git a/libc/sysdeps/linux/sparc64/soft-fp/s_scalblnl.c b/libc/sysdeps/linux/sparc64/soft-fp/s_scalblnl.c new file mode 100644 index 000000000..250a8f4f0 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/s_scalblnl.c @@ -0,0 +1,52 @@ +/* Software floating-point emulation. +   scalblnl(x, exp) +   Copyright (C) 1999-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +/* + * scalblnl (long double x, long int n) + * scalblnl(x,n) returns x* 2**n  computed by  exponent + * manipulation rather than by actually performing an + * exponentiation or a multiplication. + */ + +#include "soft-fp.h" +#include "quad.h" + +long double __scalblnl(long double arg, int exp) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); +  long double r; + +  FP_UNPACK_Q(A, arg); +  switch (A_c) +    { +    case FP_CLS_ZERO: +      return arg; +    case FP_CLS_NAN: +    case FP_CLS_INF: +      FP_HANDLE_EXCEPTIONS; +      return arg; +    } +  A_e += exp; +  FP_PACK_Q(r, A); +  FP_HANDLE_EXCEPTIONS; + +  return r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/s_scalbnl.c b/libc/sysdeps/linux/sparc64/soft-fp/s_scalbnl.c new file mode 100644 index 000000000..c686175e9 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/s_scalbnl.c @@ -0,0 +1,52 @@ +/* Software floating-point emulation. +   scalbnl(x, exp) +   Copyright (C) 1999-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Jakub Jelinek (jj@ultra.linux.cz). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +/* + * scalbnl (long double x, int n) + * scalbnl(x,n) returns x* 2**n  computed by  exponent + * manipulation rather than by actually performing an + * exponentiation or a multiplication. + */ + +#include "soft-fp.h" +#include "quad.h" + +long double __scalbnl(long double arg, int exp) +{ +  FP_DECL_EX; +  FP_DECL_Q(A); +  long double r; + +  FP_UNPACK_Q(A, arg); +  switch (A_c) +    { +    case FP_CLS_ZERO: +      return arg; +    case FP_CLS_NAN: +    case FP_CLS_INF: +      FP_HANDLE_EXCEPTIONS; +      return arg; +    } +  A_e += exp; +  FP_PACK_Q(r, A); +  FP_HANDLE_EXCEPTIONS; + +  return r; +} diff --git a/libc/sysdeps/linux/sparc64/soft-fp/sfp-machine.h b/libc/sysdeps/linux/sparc64/soft-fp/sfp-machine.h new file mode 100644 index 000000000..c03a6d376 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/sfp-machine.h @@ -0,0 +1,147 @@ +/* Machine-dependent software floating-point definitions. +   Sparc64 userland (_Q_* and _Qp_*) version. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com), +		  Jakub Jelinek (jj@ultra.linux.cz) and +		  David S. Miller (davem@redhat.com). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <fpu_control.h> +#include <fenv.h> +#include <stdlib.h> + +#define _FP_W_TYPE_SIZE		64 +#define _FP_W_TYPE		unsigned long +#define _FP_WS_TYPE		signed long +#define _FP_I_TYPE		long + +/* Helper macros for _FP_MUL_MEAT_2_120_240_double.  */ +#define _FP_MUL_MEAT_SET_FE_TZ		   			\ +do {								\ +  static fpu_control_t _fetz = _FPU_RC_DOWN;			\ +  _FPU_SETCW(_fetz);						\ +} while (0) +#ifndef _FP_MUL_MEAT_RESET_FE +#define _FP_MUL_MEAT_RESET_FE _FPU_SETCW(_fcw) +#endif + +#define _FP_MUL_MEAT_S(R,X,Y)					\ +  _FP_MUL_MEAT_1_imm(_FP_WFRACBITS_S,R,X,Y) +#define _FP_MUL_MEAT_D(R,X,Y)					\ +  _FP_MUL_MEAT_1_wide(_FP_WFRACBITS_D,R,X,Y,umul_ppmm) +#define _FP_MUL_MEAT_Q(R,X,Y)					\ +  _FP_MUL_MEAT_2_120_240_double(_FP_WFRACBITS_Q,R,X,Y,		\ +				_FP_MUL_MEAT_SET_FE_TZ,		\ +				_FP_MUL_MEAT_RESET_FE) + +#define _FP_DIV_MEAT_S(R,X,Y)	_FP_DIV_MEAT_1_imm(S,R,X,Y,_FP_DIV_HELP_imm) +#define _FP_DIV_MEAT_D(R,X,Y)	_FP_DIV_MEAT_1_udiv_norm(D,R,X,Y) +#define _FP_DIV_MEAT_Q(R,X,Y)	_FP_DIV_MEAT_2_udiv(Q,R,X,Y) + +#define _FP_NANFRAC_S		((_FP_QNANBIT_S << 1) - 1) +#define _FP_NANFRAC_D		((_FP_QNANBIT_D << 1) - 1) +#define _FP_NANFRAC_Q		((_FP_QNANBIT_Q << 1) - 1), -1 +#define _FP_NANSIGN_S		0 +#define _FP_NANSIGN_D		0 +#define _FP_NANSIGN_Q		0 + +#define _FP_KEEPNANFRACP 1 +#define _FP_QNANNEGATEDP 0 + +/* If one NaN is signaling and the other is not, + * we choose that one, otherwise we choose Y. + */ +#define _FP_CHOOSENAN(fs, wc, R, X, Y, OP)			\ +  do {								\ +    if ((_FP_FRAC_HIGH_RAW_##fs(Y) & _FP_QNANBIT_##fs)		\ +	&& !(_FP_FRAC_HIGH_RAW_##fs(X) & _FP_QNANBIT_##fs))	\ +      {								\ +	R##_s = X##_s;						\ +	_FP_FRAC_COPY_##wc(R,X);				\ +      }								\ +    else							\ +      {								\ +	R##_s = Y##_s;						\ +	_FP_FRAC_COPY_##wc(R,Y);				\ +      }								\ +    R##_c = FP_CLS_NAN;						\ +  } while (0) + +/* Obtain the current rounding mode. */ +#ifndef FP_ROUNDMODE +#define FP_ROUNDMODE	((_fcw >> 30) & 0x3) +#endif + +/* Exception flags. */ +#define FP_EX_INVALID		(1 << 4) +#define FP_EX_OVERFLOW		(1 << 3) +#define FP_EX_UNDERFLOW		(1 << 2) +#define FP_EX_DIVZERO		(1 << 1) +#define FP_EX_INEXACT		(1 << 0) + +#define _FP_TININESS_AFTER_ROUNDING 0 + +#define _FP_DECL_EX \ +  fpu_control_t _fcw __attribute__ ((unused)) = (FP_RND_NEAREST << 30) + +#define FP_INIT_ROUNDMODE					\ +do {								\ +  _FPU_GETCW(_fcw);						\ +} while (0) + +#define FP_TRAPPING_EXCEPTIONS ((_fcw >> 23) & 0x1f) +#define FP_INHIBIT_RESULTS ((_fcw >> 23) & _fex) + +/* Simulate exceptions using double arithmetics. */ +extern void __Qp_handle_exceptions(int exc); + +#define FP_HANDLE_EXCEPTIONS					\ +do {								\ +  if (!_fex)							\ +    {								\ +      /* This is the common case, so we do it inline.		\ +       * We need to clear cexc bits if any.			\ +       */							\ +      __asm__ __volatile__("fzero %%f62\n\t"			\ +			   "faddd %%f62, %%f62, %%f62"		\ +			   : : : "f62");			\ +    }								\ +  else								\ +    {								\ +      __Qp_handle_exceptions (_fex);				\ +    }								\ +} while (0) + +#define QP_HANDLE_EXCEPTIONS(_a)				\ +do {								\ +  if ((_fcw >> 23) & _fex)					\ +    {								\ +      _a;							\ +    }								\ +  else								\ +    {								\ +      _fcw = (_fcw & ~0x1fL) | (_fex << 5) | _fex;		\ +      _FPU_SETCW(_fcw);						\ +    }								\ +} while (0) + +#define QP_NO_EXCEPTIONS					\ +  __asm ("fzero %%f62\n\t"					\ +	 "faddd %%f62, %%f62, %%f62" : : : "f62") + +#define QP_CLOBBER "memory", "f52", "f54", "f56", "f58", "f60", "f62" +#define QP_CLOBBER_CC QP_CLOBBER , "cc" diff --git a/libc/sysdeps/linux/sparc64/soft-fp/single.h b/libc/sysdeps/linux/sparc64/soft-fp/single.h new file mode 100644 index 000000000..75ade9fbe --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/single.h @@ -0,0 +1,199 @@ +/* Software floating-point emulation. +   Definitions for IEEE Single Precision. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com), +		  Jakub Jelinek (jj@ultra.linux.cz), +		  David S. Miller (davem@redhat.com) and +		  Peter Maydell (pmaydell@chiark.greenend.org.uk). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 of the License, or (at your option) any later version. + +   In addition to the permissions in the GNU Lesser General Public +   License, the Free Software Foundation gives you unlimited +   permission to link the compiled version of this file into +   combinations with other programs, and to distribute those +   combinations without any restriction coming from the use of this +   file.  (The Lesser General Public License restrictions do apply in +   other respects; for example, they cover modification of the file, +   and distribution when not linked into a combine executable.) + +   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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef SOFT_FP_SINGLE_H +#define SOFT_FP_SINGLE_H	1 + +#if _FP_W_TYPE_SIZE < 32 +# error "Here's a nickel kid.  Go buy yourself a real computer." +#endif + +#define _FP_FRACTBITS_S		_FP_W_TYPE_SIZE + +#if _FP_W_TYPE_SIZE < 64 +# define _FP_FRACTBITS_DW_S	(2 * _FP_W_TYPE_SIZE) +#else +# define _FP_FRACTBITS_DW_S	_FP_W_TYPE_SIZE +#endif + +#define _FP_FRACBITS_S		24 +#define _FP_FRACXBITS_S		(_FP_FRACTBITS_S - _FP_FRACBITS_S) +#define _FP_WFRACBITS_S		(_FP_WORKBITS + _FP_FRACBITS_S) +#define _FP_WFRACXBITS_S	(_FP_FRACTBITS_S - _FP_WFRACBITS_S) +#define _FP_EXPBITS_S		8 +#define _FP_EXPBIAS_S		127 +#define _FP_EXPMAX_S		255 +#define _FP_QNANBIT_S		((_FP_W_TYPE) 1 << (_FP_FRACBITS_S-2)) +#define _FP_QNANBIT_SH_S	((_FP_W_TYPE) 1 << (_FP_FRACBITS_S-2+_FP_WORKBITS)) +#define _FP_IMPLBIT_S		((_FP_W_TYPE) 1 << (_FP_FRACBITS_S-1)) +#define _FP_IMPLBIT_SH_S	((_FP_W_TYPE) 1 << (_FP_FRACBITS_S-1+_FP_WORKBITS)) +#define _FP_OVERFLOW_S		((_FP_W_TYPE) 1 << (_FP_WFRACBITS_S)) + +#define _FP_WFRACBITS_DW_S	(2 * _FP_WFRACBITS_S) +#define _FP_WFRACXBITS_DW_S	(_FP_FRACTBITS_DW_S - _FP_WFRACBITS_DW_S) +#define _FP_HIGHBIT_DW_S	\ +  ((_FP_W_TYPE) 1 << (_FP_WFRACBITS_DW_S - 1) % _FP_W_TYPE_SIZE) + +/* The implementation of _FP_MUL_MEAT_S and _FP_DIV_MEAT_S should be +   chosen by the target machine.  */ + +typedef float SFtype __attribute__ ((mode (SF))); + +union _FP_UNION_S +{ +  SFtype flt; +  struct _FP_STRUCT_LAYOUT +  { +#if __BYTE_ORDER == __BIG_ENDIAN +    unsigned sign : 1; +    unsigned exp  : _FP_EXPBITS_S; +    unsigned frac : _FP_FRACBITS_S - (_FP_IMPLBIT_S != 0); +#else +    unsigned frac : _FP_FRACBITS_S - (_FP_IMPLBIT_S != 0); +    unsigned exp  : _FP_EXPBITS_S; +    unsigned sign : 1; +#endif +  } bits __attribute__ ((packed)); +}; + +#define FP_DECL_S(X)		_FP_DECL (1, X) +#define FP_UNPACK_RAW_S(X, val)	_FP_UNPACK_RAW_1 (S, X, (val)) +#define FP_UNPACK_RAW_SP(X, val)	_FP_UNPACK_RAW_1_P (S, X, (val)) +#define FP_PACK_RAW_S(val, X)	_FP_PACK_RAW_1 (S, (val), X) +#define FP_PACK_RAW_SP(val, X)			\ +  do						\ +    {						\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_1_P (S, (val), X);		\ +    }						\ +  while (0) + +#define FP_UNPACK_S(X, val)			\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_1 (S, X, (val));		\ +      _FP_UNPACK_CANONICAL (S, 1, X);		\ +    }						\ +  while (0) + +#define FP_UNPACK_SP(X, val)			\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_1_P (S, X, (val));		\ +      _FP_UNPACK_CANONICAL (S, 1, X);		\ +    }						\ +  while (0) + +#define FP_UNPACK_SEMIRAW_S(X, val)		\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_1 (S, X, (val));		\ +      _FP_UNPACK_SEMIRAW (S, 1, X);		\ +    }						\ +  while (0) + +#define FP_UNPACK_SEMIRAW_SP(X, val)		\ +  do						\ +    {						\ +      _FP_UNPACK_RAW_1_P (S, X, (val));		\ +      _FP_UNPACK_SEMIRAW (S, 1, X);		\ +    }						\ +  while (0) + +#define FP_PACK_S(val, X)			\ +  do						\ +    {						\ +      _FP_PACK_CANONICAL (S, 1, X);		\ +      _FP_PACK_RAW_1 (S, (val), X);		\ +    }						\ +  while (0) + +#define FP_PACK_SP(val, X)			\ +  do						\ +    {						\ +      _FP_PACK_CANONICAL (S, 1, X);		\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_1_P (S, (val), X);		\ +    }						\ +  while (0) + +#define FP_PACK_SEMIRAW_S(val, X)		\ +  do						\ +    {						\ +      _FP_PACK_SEMIRAW (S, 1, X);		\ +      _FP_PACK_RAW_1 (S, (val), X);		\ +    }						\ +  while (0) + +#define FP_PACK_SEMIRAW_SP(val, X)		\ +  do						\ +    {						\ +      _FP_PACK_SEMIRAW (S, 1, X);		\ +      if (!FP_INHIBIT_RESULTS)			\ +	_FP_PACK_RAW_1_P (S, (val), X);		\ +    }						\ +  while (0) + +#define FP_ISSIGNAN_S(X)		_FP_ISSIGNAN (S, 1, X) +#define FP_NEG_S(R, X)			_FP_NEG (S, 1, R, X) +#define FP_ADD_S(R, X, Y)		_FP_ADD (S, 1, R, X, Y) +#define FP_SUB_S(R, X, Y)		_FP_SUB (S, 1, R, X, Y) +#define FP_MUL_S(R, X, Y)		_FP_MUL (S, 1, R, X, Y) +#define FP_DIV_S(R, X, Y)		_FP_DIV (S, 1, R, X, Y) +#define FP_SQRT_S(R, X)			_FP_SQRT (S, 1, R, X) +#define _FP_SQRT_MEAT_S(R, S, T, X, Q)	_FP_SQRT_MEAT_1 (R, S, T, X, (Q)) + +#if _FP_W_TYPE_SIZE < 64 +# define FP_FMA_S(R, X, Y, Z)	_FP_FMA (S, 1, 2, R, X, Y, Z) +#else +# define FP_FMA_S(R, X, Y, Z)	_FP_FMA (S, 1, 1, R, X, Y, Z) +#endif + +#define FP_CMP_S(r, X, Y, un, ex)	_FP_CMP (S, 1, (r), X, Y, (un), (ex)) +#define FP_CMP_EQ_S(r, X, Y, ex)	_FP_CMP_EQ (S, 1, (r), X, Y, (ex)) +#define FP_CMP_UNORD_S(r, X, Y, ex)	_FP_CMP_UNORD (S, 1, (r), X, Y, (ex)) + +#define FP_TO_INT_S(r, X, rsz, rsg)	_FP_TO_INT (S, 1, (r), X, (rsz), (rsg)) +#define FP_TO_INT_ROUND_S(r, X, rsz, rsg)	\ +  _FP_TO_INT_ROUND (S, 1, (r), X, (rsz), (rsg)) +#define FP_FROM_INT_S(X, r, rs, rt)	_FP_FROM_INT (S, 1, X, (r), (rs), rt) + +#define _FP_FRAC_HIGH_S(X)	_FP_FRAC_HIGH_1 (X) +#define _FP_FRAC_HIGH_RAW_S(X)	_FP_FRAC_HIGH_1 (X) + +#if _FP_W_TYPE_SIZE < 64 +# define _FP_FRAC_HIGH_DW_S(X)	_FP_FRAC_HIGH_2 (X) +#else +# define _FP_FRAC_HIGH_DW_S(X)	_FP_FRAC_HIGH_1 (X) +#endif + +#endif /* !SOFT_FP_SINGLE_H */ diff --git a/libc/sysdeps/linux/sparc64/soft-fp/soft-fp.h b/libc/sysdeps/linux/sparc64/soft-fp/soft-fp.h new file mode 100644 index 000000000..079402e92 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/soft-fp/soft-fp.h @@ -0,0 +1,342 @@ +/* Software floating-point emulation. +   Copyright (C) 1997-2017 Free Software Foundation, Inc. +   This file is part of the GNU C Library. +   Contributed by Richard Henderson (rth@cygnus.com), +		  Jakub Jelinek (jj@ultra.linux.cz), +		  David S. Miller (davem@redhat.com) and +		  Peter Maydell (pmaydell@chiark.greenend.org.uk). + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 of the License, or (at your option) any later version. + +   In addition to the permissions in the GNU Lesser General Public +   License, the Free Software Foundation gives you unlimited +   permission to link the compiled version of this file into +   combinations with other programs, and to distribute those +   combinations without any restriction coming from the use of this +   file.  (The Lesser General Public License restrictions do apply in +   other respects; for example, they cover modification of the file, +   and distribution when not linked into a combine executable.) + +   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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef SOFT_FP_H +#define SOFT_FP_H	1 + +#include "sfp-machine.h" + +/* Allow sfp-machine to have its own byte order definitions.  */ +#ifndef __BYTE_ORDER +# ifdef _LIBC +#  include <endian.h> +# else +#  error "endianness not defined by sfp-machine.h" +# endif +#endif + +/* For unreachable default cases in switch statements over bitwise OR +   of FP_CLS_* values.  */ +#if (defined __GNUC__							\ +     && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 5))) +# define _FP_UNREACHABLE	__builtin_unreachable () +#else +# define _FP_UNREACHABLE	abort () +#endif + +#if ((defined __GNUC__							\ +      && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 6)))	\ +     || (defined __STDC_VERSION__ && __STDC_VERSION__ >= 201112L)) +# define _FP_STATIC_ASSERT(expr, msg)		\ +  _Static_assert ((expr), msg) +#else +# define _FP_STATIC_ASSERT(expr, msg)					\ +  extern int (*__Static_assert_function (void))				\ +    [!!sizeof (struct { int __error_if_negative: (expr) ? 2 : -1; })] +#endif + +/* In the Linux kernel, some architectures have a single function that +   uses different kinds of unpacking and packing depending on the +   instruction being emulated, meaning it is not readily visible to +   the compiler that variables from _FP_DECL and _FP_FRAC_DECL_* +   macros are only used in cases where they were initialized.  */ +#ifdef __KERNEL__ +# define _FP_ZERO_INIT		= 0 +#else +# define _FP_ZERO_INIT +#endif + +#define _FP_WORKBITS		3 +#define _FP_WORK_LSB		((_FP_W_TYPE) 1 << 3) +#define _FP_WORK_ROUND		((_FP_W_TYPE) 1 << 2) +#define _FP_WORK_GUARD		((_FP_W_TYPE) 1 << 1) +#define _FP_WORK_STICKY		((_FP_W_TYPE) 1 << 0) + +#ifndef FP_RND_NEAREST +# define FP_RND_NEAREST		0 +# define FP_RND_ZERO		1 +# define FP_RND_PINF		2 +# define FP_RND_MINF		3 +#endif +#ifndef FP_ROUNDMODE +# define FP_ROUNDMODE		FP_RND_NEAREST +#endif + +/* By default don't care about exceptions.  */ +#ifndef FP_EX_INVALID +# define FP_EX_INVALID		0 +#endif +#ifndef FP_EX_OVERFLOW +# define FP_EX_OVERFLOW		0 +#endif +#ifndef FP_EX_UNDERFLOW +# define FP_EX_UNDERFLOW	0 +#endif +#ifndef FP_EX_DIVZERO +# define FP_EX_DIVZERO		0 +#endif +#ifndef FP_EX_INEXACT +# define FP_EX_INEXACT		0 +#endif +#ifndef FP_EX_DENORM +# define FP_EX_DENORM		0 +#endif + +/* Sub-exceptions of "invalid".  */ +/* Signaling NaN operand.  */ +#ifndef FP_EX_INVALID_SNAN +# define FP_EX_INVALID_SNAN	0 +#endif +/* Inf * 0.  */ +#ifndef FP_EX_INVALID_IMZ +# define FP_EX_INVALID_IMZ	0 +#endif +/* fma (Inf, 0, c).  */ +#ifndef FP_EX_INVALID_IMZ_FMA +# define FP_EX_INVALID_IMZ_FMA	0 +#endif +/* Inf - Inf.  */ +#ifndef FP_EX_INVALID_ISI +# define FP_EX_INVALID_ISI	0 +#endif +/* 0 / 0.  */ +#ifndef FP_EX_INVALID_ZDZ +# define FP_EX_INVALID_ZDZ	0 +#endif +/* Inf / Inf.  */ +#ifndef FP_EX_INVALID_IDI +# define FP_EX_INVALID_IDI	0 +#endif +/* sqrt (negative).  */ +#ifndef FP_EX_INVALID_SQRT +# define FP_EX_INVALID_SQRT	0 +#endif +/* Invalid conversion to integer.  */ +#ifndef FP_EX_INVALID_CVI +# define FP_EX_INVALID_CVI	0 +#endif +/* Invalid comparison.  */ +#ifndef FP_EX_INVALID_VC +# define FP_EX_INVALID_VC	0 +#endif + +/* _FP_STRUCT_LAYOUT may be defined as an attribute to determine the +   struct layout variant used for structures where bit-fields are used +   to access specific parts of binary floating-point numbers.  This is +   required for systems where the default ABI uses struct layout with +   differences in how consecutive bit-fields are laid out from the +   default expected by soft-fp.  */ +#ifndef _FP_STRUCT_LAYOUT +# define _FP_STRUCT_LAYOUT +#endif + +#ifdef _FP_DECL_EX +# define FP_DECL_EX					\ +  int _fex = 0;						\ +  _FP_DECL_EX +#else +# define FP_DECL_EX int _fex = 0 +#endif + +/* Initialize any machine-specific state used in FP_ROUNDMODE, +   FP_TRAPPING_EXCEPTIONS or FP_HANDLE_EXCEPTIONS.  */ +#ifndef FP_INIT_ROUNDMODE +# define FP_INIT_ROUNDMODE do {} while (0) +#endif + +/* Initialize any machine-specific state used in +   FP_TRAPPING_EXCEPTIONS or FP_HANDLE_EXCEPTIONS.  */ +#ifndef FP_INIT_TRAPPING_EXCEPTIONS +# define FP_INIT_TRAPPING_EXCEPTIONS FP_INIT_ROUNDMODE +#endif + +/* Initialize any machine-specific state used in +   FP_HANDLE_EXCEPTIONS.  */ +#ifndef FP_INIT_EXCEPTIONS +# define FP_INIT_EXCEPTIONS FP_INIT_TRAPPING_EXCEPTIONS +#endif + +#ifndef FP_HANDLE_EXCEPTIONS +# define FP_HANDLE_EXCEPTIONS do {} while (0) +#endif + +/* Whether to flush subnormal inputs to zero with the same sign.  */ +#ifndef FP_DENORM_ZERO +# define FP_DENORM_ZERO 0 +#endif + +#ifndef FP_INHIBIT_RESULTS +/* By default we write the results always. +   sfp-machine may override this and e.g. +   check if some exceptions are unmasked +   and inhibit it in such a case.  */ +# define FP_INHIBIT_RESULTS 0 +#endif + +#define FP_SET_EXCEPTION(ex)				\ +  _fex |= (ex) + +#define FP_CUR_EXCEPTIONS				\ +  (_fex) + +#ifndef FP_TRAPPING_EXCEPTIONS +# define FP_TRAPPING_EXCEPTIONS 0 +#endif + +/* A file using soft-fp may define FP_NO_EXCEPTIONS before including +   soft-fp.h to indicate that, although a macro used there could raise +   exceptions, or do rounding and potentially thereby raise +   exceptions, for some arguments, for the particular arguments used +   in that file no exceptions or rounding can occur.  Such a file +   should not itself use macros relating to handling exceptions and +   rounding modes; this is only for indirect uses (in particular, in +   _FP_FROM_INT and the macros it calls).  */ +#ifdef FP_NO_EXCEPTIONS + +# undef FP_SET_EXCEPTION +# define FP_SET_EXCEPTION(ex) do {} while (0) + +# undef FP_CUR_EXCEPTIONS +# define FP_CUR_EXCEPTIONS 0 + +# undef FP_TRAPPING_EXCEPTIONS +# define FP_TRAPPING_EXCEPTIONS 0 + +# undef FP_ROUNDMODE +# define FP_ROUNDMODE FP_RND_ZERO + +# undef _FP_TININESS_AFTER_ROUNDING +# define _FP_TININESS_AFTER_ROUNDING 0 + +#endif + +/* A file using soft-fp may define FP_NO_EXACT_UNDERFLOW before +   including soft-fp.h to indicate that, although a macro used there +   could allow for the case of exact underflow requiring the underflow +   exception to be raised if traps are enabled, for the particular +   arguments used in that file no exact underflow can occur.  */ +#ifdef FP_NO_EXACT_UNDERFLOW +# undef FP_TRAPPING_EXCEPTIONS +# define FP_TRAPPING_EXCEPTIONS 0 +#endif + +#define _FP_ROUND_NEAREST(wc, X)				\ +  do								\ +    {								\ +      if ((_FP_FRAC_LOW_##wc (X) & 15) != _FP_WORK_ROUND)	\ +	_FP_FRAC_ADDI_##wc (X, _FP_WORK_ROUND);			\ +    }								\ +  while (0) + +#define _FP_ROUND_ZERO(wc, X)		(void) 0 + +#define _FP_ROUND_PINF(wc, X)				\ +  do							\ +    {							\ +      if (!X##_s && (_FP_FRAC_LOW_##wc (X) & 7))	\ +	_FP_FRAC_ADDI_##wc (X, _FP_WORK_LSB);		\ +    }							\ +  while (0) + +#define _FP_ROUND_MINF(wc, X)			\ +  do						\ +    {						\ +      if (X##_s && (_FP_FRAC_LOW_##wc (X) & 7))	\ +	_FP_FRAC_ADDI_##wc (X, _FP_WORK_LSB);	\ +    }						\ +  while (0) + +#define _FP_ROUND(wc, X)			\ +  do						\ +    {						\ +      if (_FP_FRAC_LOW_##wc (X) & 7)		\ +	{					\ +	  FP_SET_EXCEPTION (FP_EX_INEXACT);	\ +	  switch (FP_ROUNDMODE)			\ +	    {					\ +	    case FP_RND_NEAREST:		\ +	      _FP_ROUND_NEAREST (wc, X);	\ +	      break;				\ +	    case FP_RND_ZERO:			\ +	      _FP_ROUND_ZERO (wc, X);		\ +	      break;				\ +	    case FP_RND_PINF:			\ +	      _FP_ROUND_PINF (wc, X);		\ +	      break;				\ +	    case FP_RND_MINF:			\ +	      _FP_ROUND_MINF (wc, X);		\ +	      break;				\ +	    }					\ +	}					\ +    }						\ +  while (0) + +#define FP_CLS_NORMAL		0 +#define FP_CLS_ZERO		1 +#define FP_CLS_INF		2 +#define FP_CLS_NAN		3 + +#define _FP_CLS_COMBINE(x, y)	(((x) << 2) | (y)) + +#include "op-1.h" +#include "op-2.h" +#include "op-4.h" +#include "op-8.h" +#include "op-common.h" + +/* Sigh.  Silly things longlong.h needs.  */ +#define UWtype		_FP_W_TYPE +#define W_TYPE_SIZE	_FP_W_TYPE_SIZE + +typedef int QItype __attribute__ ((mode (QI))); +typedef int SItype __attribute__ ((mode (SI))); +typedef int DItype __attribute__ ((mode (DI))); +typedef unsigned int UQItype __attribute__ ((mode (QI))); +typedef unsigned int USItype __attribute__ ((mode (SI))); +typedef unsigned int UDItype __attribute__ ((mode (DI))); +#if _FP_W_TYPE_SIZE == 32 +typedef unsigned int UHWtype __attribute__ ((mode (HI))); +#elif _FP_W_TYPE_SIZE == 64 +typedef USItype UHWtype; +#endif + +#ifndef CMPtype +# define CMPtype	int +#endif + +#define SI_BITS		(__CHAR_BIT__ * (int) sizeof (SItype)) +#define DI_BITS		(__CHAR_BIT__ * (int) sizeof (DItype)) + +#ifndef umul_ppmm +# include "longlong.h" +#endif + +#endif /* !SOFT_FP_H */ diff --git a/libc/sysdeps/linux/sparc64/swapcontext.c b/libc/sysdeps/linux/sparc64/swapcontext.c new file mode 100644 index 000000000..f6dfc20ba --- /dev/null +++ b/libc/sysdeps/linux/sparc64/swapcontext.c @@ -0,0 +1,46 @@ +/* Copyright (C) 2001-2017 Free Software Foundation, Inc. +   Contributed by Jakub Jelinek <jakub@redhat.com>. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <ucontext.h> + +extern int __getcontext (ucontext_t *ucp); +extern int __setcontext (const ucontext_t *ucp, int restoremask); + +int +__swapcontext (ucontext_t *oucp, const ucontext_t *ucp) +{ +  extern void __swapcontext_ret (void); +  /* Save the current machine context to oucp.  */ +  __getcontext (oucp); +  /* Modify oucp to skip the __setcontext call on reactivation.  */ +  oucp->uc_mcontext.mc_gregs[MC_PC] = (long) __swapcontext_ret; +  oucp->uc_mcontext.mc_gregs[MC_NPC] = ((long) __swapcontext_ret) + 4; +  /* Restore the machine context in ucp.  */ +  __setcontext (ucp, 1); +  return 0; +} + +__asm__ ("							\n\ +	.text						\n\ +	.type	__swapcontext_ret, #function		\n\ +__swapcontext_ret:					\n\ +	return	%i7 + 8					\n\ +	 clr	%o0					\n\ +	.size	__swapcontext_ret, .-__swapcontext_ret	\n\ +     "); + +weak_alias (__swapcontext, swapcontext) diff --git a/libc/sysdeps/linux/sparc64/sys/procfs.h b/libc/sysdeps/linux/sparc64/sys/procfs.h new file mode 100644 index 000000000..c1dfb1980 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/sys/procfs.h @@ -0,0 +1,121 @@ +/* Copyright (C) 1996, 1997, 1999, 2000 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_PROCFS_H +#define _SYS_PROCFS_H	1 + +/* This is somehow modelled after the file of the same name on SysVr4 +   systems.  It provides a definition of the core file format for ELF +   used on Linux.  */ + +#include <features.h> +#include <signal.h> +#include <sys/time.h> +#include <sys/types.h> +#include <sys/ucontext.h> +#include <sys/user.h> +#include <bits/wordsize.h> + +__BEGIN_DECLS + +#define ELF_NGREG		38 + +typedef struct +  { +    union +      { +	unsigned long	pr_regs[32]; +	double		pr_dregs[16]; +      }			pr_fr; +    unsigned long	__unused; +    unsigned long	pr_fsr; +    unsigned char	pr_qcnt; +    unsigned char	pr_q_entrysize; +    unsigned char	pr_en; +    unsigned int	pr_q[64]; +  } elf_fpregset_t; + +typedef unsigned long elf_greg_t; +typedef elf_greg_t elf_gregset_t[ELF_NGREG]; + +struct elf_siginfo +  { +    int si_signo;			/* Signal number.  */ +    int si_code;			/* Extra code.  */ +    int si_errno;			/* Errno.  */ +  }; + +/* Definitions to generate Intel SVR4-like core files.  These mostly +   have the same names as the SVR4 types with "elf_" tacked on the +   front to prevent clashes with linux definitions, and the typedef +   forms have been avoided.  This is mostly like the SVR4 structure, +   but more Linuxy, with things that Linux does not support and which +   gdb doesn't really use excluded.  Fields present but not used are +   marked with "XXX".  */ +struct elf_prstatus +  { +    struct elf_siginfo pr_info;		/* Info associated with signal.  */ +    short int pr_cursig;		/* Current signal.  */ +    unsigned long int pr_sigpend;	/* Set of pending signals.  */ +    unsigned long int pr_sighold;	/* Set of held signals.  */ +    __pid_t pr_pid; +    __pid_t pr_ppid; +    __pid_t pr_pgrp; +    __pid_t pr_sid; +    struct timeval pr_utime;		/* User time.  */ +    struct timeval pr_stime;		/* System time.  */ +    struct timeval pr_cutime;		/* Cumulative user time.  */ +    struct timeval pr_cstime;		/* Cumulative system time.  */ +    elf_gregset_t pr_reg;		/* GP registers.  */ +    int pr_fpvalid;			/* True if math copro being used.  */ +  }; + + +#define ELF_PRARGSZ     (80)    /* Number of chars for args */ + +struct elf_prpsinfo +  { +    char pr_state;			/* Numeric process state.  */ +    char pr_sname;			/* Char for pr_state.  */ +    char pr_zomb;			/* Zombie.  */ +    char pr_nice;			/* Nice val.  */ +    unsigned long int pr_flag;		/* Flags.  */ +    unsigned short int pr_uid; +    unsigned short int pr_gid; +    int pr_pid, pr_ppid, pr_pgrp, pr_sid; +    /* Lots missing */ +    char pr_fname[16];			/* Filename of executable.  */ +    char pr_psargs[ELF_PRARGSZ];	/* Initial part of arg list.  */ +  }; + +/* Addresses.  */ +typedef void *psaddr_t; + +/* Register sets.  Linux has different names.  */ +typedef elf_gregset_t prgregset_t; +typedef elf_fpregset_t prfpregset_t; + +/* We don't have any differences between processes and threads, +   therefore have only one PID type.  */ +typedef __pid_t lwpid_t; + + +typedef struct elf_prstatus prstatus_t; +typedef struct elf_prpsinfo prpsinfo_t; + +__END_DECLS + +#endif	/* sys/procfs.h */ diff --git a/libc/sysdeps/linux/sparc64/sys/ptrace.h b/libc/sysdeps/linux/sparc64/sys/ptrace.h new file mode 100644 index 000000000..cd6675d22 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/sys/ptrace.h @@ -0,0 +1,235 @@ +/* `ptrace' debugger support interface.  Linux/SPARC version. +   Copyright (C) 1996-2014 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_PTRACE_H +#define _SYS_PTRACE_H	1 + +#include <features.h> +#include <bits/types.h> + +/* Linux/SPARC kernels up to 2.3.18 do not care much +   about what namespace polution, so use a kludge now.  */ +#undef PTRACE_GETREGS +#undef PTRACE_SETREGS +#undef PTRACE_GETFPREGS +#undef PTRACE_SETFPREGS +#undef PTRACE_READDATA +#undef PTRACE_WRITEDATA +#undef PTRACE_READTEXT +#undef PTRACE_WRITETEXT +#undef PTRACE_SUNDETACH + +__BEGIN_DECLS + +/* Type of the REQUEST argument to `ptrace.'  */ +enum __ptrace_request +{ +  /* Indicate that the process making this request should be traced. +     All signals received by this process can be intercepted by its +     parent, and its parent can use the other `ptrace' requests.  */ +  PTRACE_TRACEME = 0, +#define PT_TRACE_ME PTRACE_TRACEME + +  /* Return the word in the process's text space at address ADDR.  */ +  PTRACE_PEEKTEXT = 1, +#define PT_READ_I PTRACE_PEEKTEXT + +  /* Return the word in the process's data space at address ADDR.  */ +  PTRACE_PEEKDATA = 2, +#define PT_READ_D PTRACE_PEEKDATA + +  /* Return the word in the process's user area at offset ADDR.  */ +  PTRACE_PEEKUSER = 3, +#define PT_READ_U PTRACE_PEEKUSER + +  /* Write the word DATA into the process's text space at address ADDR.  */ +  PTRACE_POKETEXT = 4, +#define PT_WRITE_I PTRACE_POKETEXT + +  /* Write the word DATA into the process's data space at address ADDR.  */ +  PTRACE_POKEDATA = 5, +#define PT_WRITE_D PTRACE_POKEDATA + +  /* Write the word DATA into the process's user area at offset ADDR.  */ +  PTRACE_POKEUSER = 6, +#define PT_WRITE_U PTRACE_POKEUSER + +  /* Continue the process.  */ +  PTRACE_CONT = 7, +#define PT_CONTINUE PTRACE_CONT + +  /* Kill the process.  */ +  PTRACE_KILL = 8, +#define PT_KILL PTRACE_KILL + +  /* Single step the process. +     This is not supported on all machines.  */ +  PTRACE_SINGLESTEP = 9, +#define PT_STEP PTRACE_SINGLESTEP + +  /* Detach from a process attached to with PTRACE_ATTACH.  */ +  PTRACE_DETACH = 11, +#define PT_DETACH PTRACE_DETACH + +  /* This define is needed for older programs which were +     trying to work around sparc-linux ptrace nastiness.  */ +#define PTRACE_SUNDETACH PTRACE_DETACH + +  /* Attach to a process that is already running. */ +  PTRACE_ATTACH = 16, +#define PT_ATTACH PTRACE_ATTACH + +  /* Write several bytes at a time. */ +  PTRACE_WRITEDATA = 17, +#define PTRACE_WRITEDATA PTRACE_WRITEDATA + +  /* Read several bytes at a time. */ +  PTRACE_READTEXT = 18, +#define PTRACE_READTEXT PTRACE_READTEXT +#define PTRACE_READDATA PTRACE_READTEXT + +  /* Write several bytes at a time. */ +  PTRACE_WRITETEXT = 19, +#define PTRACE_WRITETEXT PTRACE_WRITETEXT + +  /* Get all general purpose registers used by a processes. +     This is not supported on all machines.  */ +   PTRACE_GETREGS = 22, +#define PT_GETREGS PTRACE_GETREGS + +  /* Set all general purpose registers used by a processes. +     This is not supported on all machines.  */ +   PTRACE_SETREGS = 23, +#define PT_SETREGS PTRACE_SETREGS + +  /* Continue and stop at the next (return from) syscall.  */ +  PTRACE_SYSCALL = 24, +#define PTRACE_SYSCALL PTRACE_SYSCALL + +  /* Get all floating point registers used by a processes. +     This is not supported on all machines.  */ +   PTRACE_GETFPREGS = 25, +#define PT_GETFPREGS PTRACE_GETFPREGS + +  /* Set all floating point registers used by a processes. +     This is not supported on all machines.  */ +   PTRACE_SETFPREGS = 26, +#define PT_SETFPREGS PTRACE_SETFPREGS + +  /* Set ptrace filter options.  */ +  PTRACE_SETOPTIONS = 0x4200, +#define PT_SETOPTIONS PTRACE_SETOPTIONS + +  /* Get last ptrace message.  */ +  PTRACE_GETEVENTMSG = 0x4201, +#define PT_GETEVENTMSG PTRACE_GETEVENTMSG + +  /* Get siginfo for process.  */ +  PTRACE_GETSIGINFO = 0x4202, +#define PT_GETSIGINFO PTRACE_GETSIGINFO + +  /* Set new siginfo for process.  */ +  PTRACE_SETSIGINFO = 0x4203, +#define PT_SETSIGINFO PTRACE_SETSIGINFO + +  /* Get register content.  */ +  PTRACE_GETREGSET = 0x4204, +#define PTRACE_GETREGSET PTRACE_GETREGSET + +  /* Set register content.  */ +  PTRACE_SETREGSET = 0x4205, +#define PTRACE_SETREGSET PTRACE_SETREGSET + +  /* Like PTRACE_ATTACH, but do not force tracee to trap and do not affect +     signal or group stop state.  */ +  PTRACE_SEIZE = 0x4206, +#define PTRACE_SEIZE PTRACE_SEIZE + +  /* Trap seized tracee.  */ +  PTRACE_INTERRUPT = 0x4207, +#define PTRACE_INTERRUPT PTRACE_INTERRUPT + +  /* Wait for next group event.  */ +  PTRACE_LISTEN = 0x4208, +#define PTRACE_LISTEN PTRACE_LISTEN + +  PTRACE_PEEKSIGINFO = 0x4209 +#define PTRACE_PEEKSIGINFO PTRACE_PEEKSIGINFO +}; + + +/* Flag for PTRACE_LISTEN.  */ +enum __ptrace_flags +{ +  PTRACE_SEIZE_DEVEL = 0x80000000 +}; + +/* Options set using PTRACE_SETOPTIONS.  */ +enum __ptrace_setoptions +{ +  PTRACE_O_TRACESYSGOOD	= 0x00000001, +  PTRACE_O_TRACEFORK	= 0x00000002, +  PTRACE_O_TRACEVFORK   = 0x00000004, +  PTRACE_O_TRACECLONE	= 0x00000008, +  PTRACE_O_TRACEEXEC	= 0x00000010, +  PTRACE_O_TRACEVFORKDONE = 0x00000020, +  PTRACE_O_TRACEEXIT	= 0x00000040, +  PTRACE_O_TRACESECCOMP = 0x00000080, +  PTRACE_O_EXITKILL	= 0x00100000, +  PTRACE_O_MASK		= 0x001000ff +}; + +/* Wait extended result codes for the above trace options.  */ +enum __ptrace_eventcodes +{ +  PTRACE_EVENT_FORK	= 1, +  PTRACE_EVENT_VFORK	= 2, +  PTRACE_EVENT_CLONE	= 3, +  PTRACE_EVENT_EXEC	= 4, +  PTRACE_EVENT_VFORK_DONE = 5, +  PTRACE_EVENT_EXIT	= 6, +  PTRACE_EVENT_SECCOMP  = 7 +}; + +/* Arguments for PTRACE_PEEKSIGINFO.  */ +struct __ptrace_peeksiginfo_args +{ +  __uint64_t off;	/* From which siginfo to start.  */ +  __uint32_t flags;	/* Flags for peeksiginfo.  */ +  __int32_t nr;		/* How many siginfos to take.  */ +}; + +enum __ptrace_peeksiginfo_flags +{ +  /* Read signals from a shared (process wide) queue.  */ +  PTRACE_PEEKSIGINFO_SHARED = (1 << 0) +}; + +/* Perform process tracing functions.  REQUEST is one of the values +   above, and determines the action to be taken. +   For all requests except PTRACE_TRACEME, PID specifies the process to be +   traced. + +   PID and the other arguments described above for the various requests should +   appear (those that are used for the particular request) as: +     pid_t PID, void *ADDR, int DATA, void *ADDR2 +   after REQUEST.  */ +extern long int ptrace (enum __ptrace_request __request, ...) __THROW; + +__END_DECLS + +#endif /* _SYS_PTRACE_H */ diff --git a/libc/sysdeps/linux/sparc64/sys/ucontext.h b/libc/sysdeps/linux/sparc64/sys/ucontext.h new file mode 100644 index 000000000..879bd7c58 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/sys/ucontext.h @@ -0,0 +1,198 @@ +/* Copyright (C) 1998-2017 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_UCONTEXT_H +#define _SYS_UCONTEXT_H	1 + +#include <features.h> +#include <signal.h> + +#define MC_TSTATE	0 +#define MC_PC		1 +#define MC_NPC		2 +#define MC_Y		3 +#define MC_G1		4 +#define MC_G2		5 +#define MC_G3		6 +#define MC_G4		7 +#define MC_G5		8 +#define MC_G6		9 +#define MC_G7		10 +#define MC_O0		11 +#define MC_O1		12 +#define MC_O2		13 +#define MC_O3		14 +#define MC_O4		15 +#define MC_O5		16 +#define MC_O6		17 +#define MC_O7		18 +#define MC_NGREG	19 + +typedef unsigned long mc_greg_t; +typedef mc_greg_t mc_gregset_t[MC_NGREG]; + +#define MC_MAXFPQ	16 +struct mc_fq { +	unsigned long	*mcfq_addr; +	unsigned int	mcfq_insn; +}; + +struct mc_fpu { +	union { +		unsigned int	sregs[32]; +		unsigned long	dregs[32]; +		long double	qregs[16]; +	} mcfpu_fregs; +	unsigned long	mcfpu_fsr; +	unsigned long	mcfpu_fprs; +	unsigned long	mcfpu_gsr; +	struct mc_fq	*mcfpu_fq; +	unsigned char	mcfpu_qcnt; +	unsigned char	mcfpu_qentsz; +	unsigned char	mcfpu_enab; +}; +typedef struct mc_fpu mc_fpu_t; + +typedef struct { +	mc_gregset_t	mc_gregs; +	mc_greg_t	mc_fp; +	mc_greg_t	mc_i7; +	mc_fpu_t	mc_fpregs; +} mcontext_t; + +typedef struct ucontext { +	struct ucontext		*uc_link; +	unsigned long		uc_flags; +	unsigned long		__uc_sigmask; +	mcontext_t		uc_mcontext; +	stack_t			uc_stack; +	__sigset_t		uc_sigmask; +} ucontext_t; + +/* + * Location of the users' stored registers relative to R0. + * Usage is as an index into a gregset_t array or as u.u_ar0[XX]. + */ +#define REG_PSR (0) +#define REG_PC  (1) +#define REG_nPC (2) +#define REG_Y   (3) +#define REG_G1  (4) +#define REG_G2  (5) +#define REG_G3  (6) +#define REG_G4  (7) +#define REG_G5  (8) +#define REG_G6  (9) +#define REG_G7  (10) +#define REG_O0  (11) +#define REG_O1  (12) +#define REG_O2  (13) +#define REG_O3  (14) +#define REG_O4  (15) +#define REG_O5  (16) +#define REG_O6  (17) +#define REG_O7  (18) + +/* + * A gregset_t is defined as an array type for compatibility with the reference + * source. This is important due to differences in the way the C language + * treats arrays and structures as parameters. + * + * Note that NGREG is really (sizeof (struct regs) / sizeof (greg_t)), + * but that the ABI defines it absolutely to be 21 (resp. 19). + */ + + +#define REG_ASI	(19) +#define REG_FPRS (20) + +#define NGREG   21 +typedef long greg_t; +typedef greg_t  gregset_t[NGREG]; + +/* + * The following structures define how a register window can appear on the + * stack. This structure is available (when required) through the `gwins' + * field of an mcontext (nested within ucontext). SPARC_MAXWINDOW is the + * maximum number of outstanding regiters window defined in the SPARC + * architecture (*not* implementation). + */ +#define SPARC_MAXREGWINDOW	31	/* max windows in SPARC arch. */ +struct  rwindow +  { +    greg_t rw_local[8];			/* locals */ +    greg_t rw_in[8];			/* ins */ +  }; + +#define rw_fp   rw_in[6]		/* frame pointer */ +#define rw_rtn  rw_in[7]		/* return address */ + +typedef struct gwindows +  { +    int            wbcnt; +    int           *spbuf[SPARC_MAXREGWINDOW]; +    struct rwindow wbuf[SPARC_MAXREGWINDOW]; +  } gwindows_t; + +/* + * Floating point definitions. + */ + +#define MAXFPQ	16	/* max # of fpu queue entries currently supported */ + +/* + * struct fq defines the minimal format of a floating point instruction queue + * entry. The size of entries in the floating point queue are implementation + * dependent. The union FQu is guarenteed to be the first field in any ABI + * conformant system implementation. Any additional fields provided by an + * implementation should not be used applications designed to be ABI conformant. */ + +struct fpq +  { +    unsigned long *fpq_addr;		/* address */ +    unsigned long fpq_instr;		/* instruction */ +  }; + +struct fq +  { +    union				/* FPU inst/addr queue */ +      { +        double whole; +        struct fpq fpq; +      } FQu; +  }; + +#define FPU_REGS_TYPE           unsigned +#define FPU_DREGS_TYPE          unsigned long long +#define V7_FPU_FSR_TYPE         unsigned +#define V9_FPU_FSR_TYPE         unsigned long long +#define V9_FPU_FPRS_TYPE        unsigned + +typedef struct fpu +  { +    union {				/* FPU floating point regs */ +      unsigned		fpu_regs[32];	/* 32 singles */ +      double            fpu_dregs[32];	/* 32 doubles */ +      long double	fpu_qregs[16];  /* 16 quads */ +    } fpu_fr; +    struct fq       *fpu_q;		/* ptr to array of FQ entries */ +    unsigned long   fpu_fsr;		/* FPU status register */ +    unsigned char   fpu_qcnt;		/* # of entries in saved FQ */ +    unsigned char   fpu_q_entrysize;	/* # of bytes per FQ entry */ +    unsigned char   fpu_en;		/* flag signifying fpu in use */ +  } fpregset_t; + +#endif /* sys/ucontext.h */ diff --git a/libc/sysdeps/linux/sparc64/sys/user.h b/libc/sysdeps/linux/sparc64/sys/user.h new file mode 100644 index 000000000..905f43b75 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/sys/user.h @@ -0,0 +1,83 @@ +/* Copyright (C) 2003 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _SYS_USER_H +#define _SYS_USER_H	1 + +struct sunos_regs +{ +  unsigned int psr, pc, npc, y; +  unsigned int regs[15]; +}; + +struct sunos_fpqueue +{ +  unsigned int *addr; +  unsigned int inst; +}; + +struct sunos_fp +{ +  union +    { +      unsigned int regs[32]; +      double reg_dbls[16]; +    } fregs; +  unsigned int fsr; +  unsigned int flags; +  unsigned int extra; +  unsigned int fpq_count; +  struct sunos_fpqueue fpq[16]; +}; + +struct sunos_fpu +{ +  struct sunos_fp fpstatus; +}; + +/* The SunOS core file header layout. */ +struct user { +  unsigned int magic; +  unsigned int len; +  struct sunos_regs regs; +  struct +    { +      unsigned char a_dynamic :1; +      unsigned char a_toolversion :7; +      unsigned char a_machtype; +      unsigned short a_info; +      unsigned int a_text; +      unsigned int a_data; +      unsigned int a_bss; +      unsigned int a_syms; +      unsigned int a_entry; +      unsigned int a_trsize; +      unsigned int a_drsize; +    } uexec; +  int           signal; +  size_t        u_tsize; +  size_t        u_dsize; +  size_t        u_ssize; +  char          u_comm[17]; +  struct sunos_fpu fpu; +  unsigned int  sigcode; +}; + +#define NBPG			0x2000 +#define UPAGES			1 +#define SUNOS_CORE_MAGIC	0x080456 + +#endif diff --git a/libc/sysdeps/linux/sparc64/syscall.S b/libc/sysdeps/linux/sparc64/syscall.S new file mode 100644 index 000000000..42c3d2827 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/syscall.S @@ -0,0 +1,38 @@ +/* Copyright (C) 1997-2017 Free Software Foundation, Inc. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <sysdep.h> +	.text + +	.globl	__syscall_error +ENTRY(syscall) +	mov	%o0,%g1 +	mov	%o1,%o0 +	mov	%o2,%o1 +	mov	%o3,%o2 +	mov	%o4,%o3 +	mov	%o5,%o4 +	ldx	[%sp + STACK_BIAS + 176],%o5 + +	ta	0x6d + +	bcc,pt	%xcc, 1f +	 mov	%o7, %g1 +	call	__syscall_error +	 mov	%g1, %o7 +1:	retl +	 nop +END(syscall) diff --git a/libc/sysdeps/linux/sparc64/sysdep.h b/libc/sysdeps/linux/sparc64/sysdep.h new file mode 100644 index 000000000..31008c34b --- /dev/null +++ b/libc/sysdeps/linux/sparc64/sysdep.h @@ -0,0 +1,139 @@ +/* Copyright (C) 1997-2017 Free Software Foundation, Inc. +   Contributed by Richard Henderson <richard@gnu.ai.mit.edu>, 1997. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#ifndef _LINUX_SPARC64_SYSDEP_H +#define _LINUX_SPARC64_SYSDEP_H 1 + +#include <common/sysdep.h> + +#undef SYS_ify +#define SYS_ify(syscall_name) __NR_##syscall_name + +#ifdef __ASSEMBLER__ + +#define SPARC_PIC_THUNK(reg)						\ +	.ifndef __sparc_get_pc_thunk.reg;				\ +	.section .text.__sparc_get_pc_thunk.reg,"axG",@progbits,__sparc_get_pc_thunk.reg,comdat; \ +	.align	 32;							\ +	.weak	 __sparc_get_pc_thunk.reg;				\ +	.hidden	 __sparc_get_pc_thunk.reg;				\ +	.type	 __sparc_get_pc_thunk.reg, #function;			\ +__sparc_get_pc_thunk.reg:		   				\ +	jmp	%o7 + 8;						\ +	 add	%o7, %reg, %##reg;					\ +	.previous;							\ +	.endif; + +/* The "-4" and "+4" offsets against _GLOBAL_OFFSET_TABLE_ are +   critical since they represent the offset from the thunk call to the +   instruction containing the _GLOBAL_OFFSET_TABLE_ reference. +   Therefore these instructions cannot be moved around without +   appropriate adjustments to those offsets. + +   Furthermore, these expressions are special in another regard.  When +   the assembler sees a reference to _GLOBAL_OFFSET_TABLE_ inside of +   a %hi() or %lo(), it emits a PC-relative relocation.  This causes +   R_SPARC_HI22 to turn into R_SPARC_PC22, and R_SPARC_LO10 to turn into +   R_SPARC_PC10, respectively. + +   Even when v9 we use a call sequence instead of using "rd %pc" because +   RDPC is extremely expensive and incurs a full pipeline flush.  */ + +#define SPARC_PIC_THUNK_CALL(reg)					\ +	sethi	%hi(_GLOBAL_OFFSET_TABLE_-4), %##reg;			\ +	call	__sparc_get_pc_thunk.reg;				\ +	 or	%##reg, %lo(_GLOBAL_OFFSET_TABLE_+4), %##reg; + +#define SETUP_PIC_REG(reg)						\ +	SPARC_PIC_THUNK(reg)						\ +	SPARC_PIC_THUNK_CALL(reg) + +#define SETUP_PIC_REG_LEAF(reg, tmp)					\ +	SPARC_PIC_THUNK(reg)						\ +	mov	%o7, %##tmp;		      				\ +	SPARC_PIC_THUNK_CALL(reg);					\ +	mov	%##tmp, %o7; + + +#define LOADSYSCALL(x) mov __NR_##x, %g1 + +#define ENTRY(name)                 \ +    .align 4;                       \ +    .global C_SYMBOL_NAME(name);    \ +    .type   name, @function;        \ +C_LABEL(name)                       \ +    cfi_startproc; + +#undef END +#define END(name)                   \ +    cfi_endproc;                    \ +    .size name, . - name + +#define LOC(name) .L##name + +#undef PSEUDO +#define PSEUDO(name, syscall_name, args)	\ +	.text;					\ +ENTRY(name);					\ +	LOADSYSCALL(syscall_name);		\ +	ta		0x6d;			\ +	bcc,pt		%xcc, 1f;		\ +	 nop;					\ +	SYSCALL_ERROR_HANDLER			\ +1: + +#undef PSEUDO_NOERRNO +#define	PSEUDO_NOERRNO(name, syscall_name, args)\ +	.text;					\ +ENTRY(name);					\ +	LOADSYSCALL(syscall_name);		\ +	ta		0x6d; + +#undef PSEUDO_ERRVAL +#define	PSEUDO_ERRVAL(name, syscall_name, args) \ +	.text;					\ +ENTRY(name);					\ +	LOADSYSCALL(syscall_name);		\ +	ta		0x6d; + +#undef PSEUDO_END +#define PSEUDO_END(name)			\ +	END(name) + +#ifndef __PIC__ +# define SYSCALL_ERROR_HANDLER			\ +	mov	%o7, %g1;			\ +	call	__syscall_error;		\ +	 mov	%g1, %o7; +#else +# define SYSCALL_ERROR_HANDLER		\ +0:	SETUP_PIC_REG_LEAF(o2,g1)	\ +	sethi	%gdop_hix22(errno), %g1;\ +	xor	%g1, %gdop_lox10(errno), %g1;\ +	ldx	[%o2 + %g1], %g1, %gdop(errno);\ +	st	%o0, [%g1];		\ +	jmp	%o7 + 8;		\ +	 mov	-1, %o0; +#endif	/* PIC */ + +#endif	/* __ASSEMBLER__ */ + +/* This is the offset from the %sp to the backing store above the +   register windows.  So if you poke stack memory directly you add this.  */ +#define STACK_BIAS	2047 + +#endif diff --git a/libc/sysdeps/linux/sparc64/ucontext_i.sym b/libc/sysdeps/linux/sparc64/ucontext_i.sym new file mode 100644 index 000000000..99788e517 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/ucontext_i.sym @@ -0,0 +1,6 @@ +UC_LINK		0 +__UC_SIGMASK	16 +UC_M_PC		40 +UC_M_NPC	48 +UC_SIGMASK	536 +SIGMASK_WORDS	16 diff --git a/libc/sysdeps/linux/sparc64/vfork.S b/libc/sysdeps/linux/sparc64/vfork.S new file mode 100644 index 000000000..0e44c2468 --- /dev/null +++ b/libc/sysdeps/linux/sparc64/vfork.S @@ -0,0 +1,36 @@ +/* Copyright (C) 2004-2017 Free Software Foundation, Inc. +   Contributed by Jakub Jelinek <jakub@redhat.com>, 2004. + +   The GNU C Library is free software; you can redistribute it and/or +   modify it under the terms of the GNU Lesser General Public +   License as published by the Free Software Foundation; either +   version 2.1 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 +   Lesser General Public License for more details. + +   You should have received a copy of the GNU Lesser General Public +   License along with the GNU C Library; if not, see +   <http://www.gnu.org/licenses/>.  */ + +#include <sysdep.h> + +	.text +	.globl	__syscall_error +ENTRY(__vfork) +	LOADSYSCALL(vfork) +	ta	0x6d +	bcc,pt	%xcc, 2f +	 mov	%o7, %g1 +	call	__syscall_error +	 mov	%g1, %o7 +2:	sub	%o1, 1, %o1 +	andcc	%o0, %o1, %o0 +1:	retl +	 nop +END(__vfork) + +weak_alias(__vfork,vfork) +libc_hidden_def(vfork) | 
