aboutsummaryrefslogtreecommitdiff
path: root/src/libmach/fpformat.c
blob: c76cca3ee7c2465cb89598cd84db2841d5d51723 (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
#include <u.h>
#include <libc.h>
#include <bio.h>
#include <mach.h>

/*
 *	Format floating point registers
 *
 *	Register codes in format field:
 *	'X' - print as 32-bit hexadecimal value
 *	'F' - 64-bit double register when modif == 'F'; else 32-bit single reg
 *	'f' - 32-bit ieee float
 *	'8' - big endian 80-bit ieee extended float
 *	'3' - little endian 80-bit ieee extended float with hole in bytes 8&9
 */
int
fpformat(Map *map, Regdesc *rp, char *buf, uint n, uint modif)
{
	char reg[12];
	u32int r;

	switch(rp->format)
	{
	case 'X':
		if (get4(map, rp->offset, &r) < 0)
			return -1;
		snprint(buf, n, "%lux", r);
		break;
	case 'F':	/* first reg of double reg pair */
		if (modif == 'F')
		if ((rp->format=='F') || (((rp+1)->flags&RFLT) && (rp+1)->format == 'f')) {
			if (get1(map, rp->offset, (uchar *)reg, 8) < 0)
				return -1;
			mach->ftoa64(buf, n, reg);
			if (rp->format == 'F')
				return 1;
			return 2;
		}
			/* treat it like 'f' */
		if (get1(map, rp->offset, (uchar *)reg, 4) < 0)
			return -1;
		mach->ftoa32(buf, n, reg);
		break;
	case 'f':	/* 32 bit float */
		if (get1(map, rp->offset, (uchar *)reg, 4) < 0)
			return -1;
		mach->ftoa32(buf, n, reg);
		break;
	case '3':	/* little endian ieee 80 with hole in bytes 8&9 */
		if (get1(map, rp->offset, (uchar *)reg, 10) < 0)
			return -1;
		memmove(reg+10, reg+8, 2);	/* open hole */
		memset(reg+8, 0, 2);		/* fill it */
		leieeeftoa80(buf, n, reg);
		break;
	case '8':	/* big-endian ieee 80 */
		if (get1(map, rp->offset, (uchar *)reg, 10) < 0)
			return -1;
		beieeeftoa80(buf, n, reg);
		break;
	default:	/* unknown */
		break;
	}
	return 1;
}