summaryrefslogtreecommitdiff
path: root/gdb/symm-tdep.c
blob: 79d9431d393da4efc23023da13354ec829d9504e (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
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
/* Sequent Symmetry target interface, for GDB.
   Copyright (C) 1986, 1987, 1989, 1991, 1994 Free Software Foundation, Inc.

   This file is part of GDB.

   This program is free software; you can redistribute it and/or modify
   it under the terms of the GNU General Public License as published by
   the Free Software Foundation; either version 2 of the License, or
   (at your option) any later version.

   This program is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.

   You should have received a copy of the GNU General Public License
   along with this program; if not, write to the Free Software
   Foundation, Inc., 59 Temple Place - Suite 330,
   Boston, MA 02111-1307, USA.  */

/* many 387-specific items of use taken from i386-dep.c */

#include "defs.h"
#include "frame.h"
#include "inferior.h"
#include "symtab.h"

#include <signal.h>
#include <sys/param.h>
#include <sys/user.h>
#include <sys/dir.h>
#include <sys/ioctl.h>
#include "gdb_stat.h"
#include "gdbcore.h"
#include <fcntl.h>

void
symmetry_extract_return_value (struct type *type, char *regbuf, char *valbuf)
{
  union
    {
      double d;
      int l[2];
    }
  xd;
  struct minimal_symbol *msymbol;
  float f;

  if (TYPE_CODE_FLT == TYPE_CODE (type))
    {
      msymbol = lookup_minimal_symbol ("1167_flt", NULL, NULL);
      if (msymbol != NULL)
	{
	  /* found "1167_flt" means 1167, %fp2-%fp3 */
	  /* float & double; 19= %fp2, 20= %fp3 */
	  /* no single precision on 1167 */
	  xd.l[1] = *((int *) &regbuf[REGISTER_BYTE (19)]);
	  xd.l[0] = *((int *) &regbuf[REGISTER_BYTE (20)]);
	  switch (TYPE_LENGTH (type))
	    {
	    case 4:
	      /* FIXME: broken for cross-debugging.  */
	      f = (float) xd.d;
	      memcpy (valbuf, &f, TYPE_LENGTH (type));
	      break;
	    case 8:
	      /* FIXME: broken for cross-debugging.  */
	      memcpy (valbuf, &xd.d, TYPE_LENGTH (type));
	      break;
	    default:
	      error ("Unknown floating point size");
	      break;
	    }
	}
      else
	{
	  /* 387 %st(0), gcc uses this */
	  i387_to_double (((int *) &regbuf[REGISTER_BYTE (3)]),
			  &xd.d);
	  switch (TYPE_LENGTH (type))
	    {
	    case 4:		/* float */
	      f = (float) xd.d;
	      /* FIXME: broken for cross-debugging.  */
	      memcpy (valbuf, &f, 4);
	      break;
	    case 8:		/* double */
	      /* FIXME: broken for cross-debugging.  */
	      memcpy (valbuf, &xd.d, 8);
	      break;
	    default:
	      error ("Unknown floating point size");
	      break;
	    }
	}
    }
  else
    {
      memcpy (valbuf, regbuf, TYPE_LENGTH (type));
    }
}