summaryrefslogtreecommitdiff
path: root/ports/sysdeps/alpha/soft-fp/local-soft-fp.h
blob: e93a2ad06493089ba4856d46671400b554d7c58a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
#include <stdlib.h>
#include <soft-fp.h>
#include <quad.h>

/* Helpers for the Ots functions which receive long double arguments
   in two integer registers, and return values in $16+$17.  */

#undef _FP_UNPACK_RAW_2
#define _FP_UNPACK_RAW_2(fs, X, val)                    \
  do {                                                  \
    union _FP_UNION_##fs _flo;				\
    _flo.longs.a = val##l;				\
    _flo.longs.b = val##h;				\
    X##_f0 = _flo.bits.frac0;				\
    X##_f1 = _flo.bits.frac1;				\
    X##_e  = _flo.bits.exp;				\
    X##_s  = _flo.bits.sign;				\
  } while (0)

#undef _FP_PACK_RAW_2
#define _FP_PACK_RAW_2(fs, val, X)                      \
  do {                                                  \
    union _FP_UNION_##fs _flo;				\
    _flo.bits.frac0 = X##_f0;				\
    _flo.bits.frac1 = X##_f1;				\
    _flo.bits.exp   = X##_e;				\
    _flo.bits.sign  = X##_s;				\
    val##l = _flo.longs.a;				\
    val##h = _flo.longs.b;				\
  } while (0)

#define FP_DECL_RETURN(X) \
  long X##l, X##h

/* ??? We don't have a real way to tell the compiler that we're wanting
   to return values in $16+$17.  Instead use a volatile asm to make sure
   that the values are live, and just hope that nothing kills the values
   in between here and the end of the function.  */
#define FP_RETURN(X)				\
do {						\
  register long r16 __asm__("16") = X##l;	\
  register long r17 __asm__("17") = X##h;	\
  asm volatile ("" : : "r"(r16), "r"(r17));	\
} while (0)