|
|
|
#include <stddef.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
|
|
|
|
#include "console.h"
|
|
|
|
|
|
|
|
#define MSR_FP 0x2000
|
|
|
|
#define MSR_FE0 0x800
|
|
|
|
#define MSR_FE1 0x100
|
|
|
|
|
|
|
|
extern int trapit(long arg, int (*func)(long));
|
|
|
|
|
|
|
|
#define SRR0 26
|
|
|
|
#define SRR1 27
|
|
|
|
|
|
|
|
static inline unsigned long mfspr(int sprnum)
|
|
|
|
{
|
|
|
|
long val;
|
|
|
|
|
|
|
|
__asm__ volatile("mfspr %0,%1" : "=r" (val) : "i" (sprnum));
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void mtspr(int sprnum, unsigned long val)
|
|
|
|
{
|
|
|
|
__asm__ volatile("mtspr %0,%1" : : "i" (sprnum), "r" (val));
|
|
|
|
}
|
|
|
|
|
|
|
|
void disable_fp(void)
|
|
|
|
{
|
|
|
|
unsigned long msr;
|
|
|
|
|
|
|
|
__asm__("mfmsr %0" : "=r" (msr));
|
|
|
|
msr &= ~(MSR_FP | MSR_FE0 | MSR_FE1);
|
|
|
|
__asm__("mtmsrd %0" : : "r" (msr));
|
|
|
|
}
|
|
|
|
|
|
|
|
void enable_fp(void)
|
|
|
|
{
|
|
|
|
unsigned long msr;
|
|
|
|
|
|
|
|
__asm__("mfmsr %0" : "=r" (msr));
|
|
|
|
msr |= MSR_FP;
|
|
|
|
__asm__("mtmsrd %0" : : "r" (msr));
|
|
|
|
}
|
|
|
|
|
|
|
|
void print_string(const char *str)
|
|
|
|
{
|
|
|
|
for (; *str; ++str)
|
|
|
|
putchar(*str);
|
|
|
|
}
|
|
|
|
|
|
|
|
void print_hex(unsigned long val, int ndigits)
|
|
|
|
{
|
|
|
|
int i, x;
|
|
|
|
|
|
|
|
for (i = (ndigits - 1) * 4; i >= 0; i -= 4) {
|
|
|
|
x = (val >> i) & 0xf;
|
|
|
|
if (x >= 10)
|
|
|
|
putchar(x + 'a' - 10);
|
|
|
|
else
|
|
|
|
putchar(x + '0');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// i < 100
|
|
|
|
void print_test_number(int i)
|
|
|
|
{
|
|
|
|
print_string("test ");
|
|
|
|
putchar(48 + i/10);
|
|
|
|
putchar(48 + i%10);
|
|
|
|
putchar(':');
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned long foo = 0x3ff8000000000000ul;
|
|
|
|
unsigned long foow;
|
|
|
|
int fooi = -76543;
|
|
|
|
int fooiw;
|
|
|
|
|
|
|
|
int do_fp_op(long arg)
|
|
|
|
{
|
|
|
|
switch (arg) {
|
|
|
|
case 0:
|
|
|
|
__asm__("lfd 31,0(%0)" : : "b" (&foo));
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
__asm__("stfd 31,0(%0)" : : "b" (&foow) : "memory");
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
__asm__("lfd 30,0(%0); stfd 30,0(%1)"
|
|
|
|
: : "b" (&foo), "b" (&foow) : "memory");
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
__asm__("lfiwax 29,0,%0; stfd 29,0(%1)"
|
|
|
|
: : "r" (&fooi), "b" (&foow) : "memory");
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
__asm__("lfiwzx 28,0,%0; stfd 28,0(%1)"
|
|
|
|
: : "r" (&fooi), "b" (&foow) : "memory");
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
__asm__("lfdx 27,0,%0; stfiwx 27,0,%1"
|
|
|
|
: : "r" (&foow), "r" (&fooiw) : "memory");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int fpu_test_1(void)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
disable_fp();
|
|
|
|
/* these should give a FP unavailable exception */
|
|
|
|
ret = trapit(0, do_fp_op);
|
|
|
|
if (ret != 0x800)
|
|
|
|
return 1;
|
|
|
|
ret = trapit(1, do_fp_op);
|
|
|
|
if (ret != 0x800)
|
|
|
|
return 2;
|
|
|
|
enable_fp();
|
|
|
|
/* these should succeed */
|
|
|
|
ret = trapit(0, do_fp_op);
|
|
|
|
if (ret)
|
|
|
|
return ret | 3;
|
|
|
|
ret = trapit(1, do_fp_op);
|
|
|
|
if (ret)
|
|
|
|
return ret | 4;
|
|
|
|
if (foow != foo)
|
|
|
|
return 5;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int fpu_test_2(void)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
enable_fp();
|
|
|
|
foow = ~0;
|
|
|
|
ret = trapit(2, do_fp_op);
|
|
|
|
if (ret)
|
|
|
|
return ret | 1;
|
|
|
|
if (foow != foo)
|
|
|
|
return 2;
|
|
|
|
foow = ~0;
|
|
|
|
ret = trapit(3, do_fp_op);
|
|
|
|
if (ret)
|
|
|
|
return ret | 3;
|
|
|
|
if (foow != fooi)
|
|
|
|
return 4;
|
|
|
|
foow = ~0;
|
|
|
|
ret = trapit(4, do_fp_op);
|
|
|
|
if (ret)
|
|
|
|
return ret | 5;
|
|
|
|
if (foow != (unsigned int)fooi)
|
|
|
|
return 6;
|
|
|
|
ret = trapit(5, do_fp_op);
|
|
|
|
if (ret)
|
|
|
|
return ret | 7;
|
|
|
|
if (fooiw != fooi)
|
|
|
|
return 8;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct sp_dp_equiv {
|
|
|
|
unsigned int sp;
|
|
|
|
unsigned long dp;
|
|
|
|
} sp_dp_equiv[] = {
|
|
|
|
{ 0, 0 },
|
|
|
|
{ 0x80000000, 0x8000000000000000 },
|
|
|
|
{ 0x7f800000, 0x7ff0000000000000 },
|
|
|
|
{ 0xff800000, 0xfff0000000000000 },
|
|
|
|
{ 0x7f812345, 0x7ff02468a0000000 },
|
|
|
|
{ 0x456789ab, 0x40acf13560000000 },
|
|
|
|
{ 0x12345678, 0x3a468acf00000000 },
|
|
|
|
{ 0x00400000, 0x3800000000000000 },
|
|
|
|
{ 0x00200000, 0x37f0000000000000 },
|
|
|
|
{ 0x00000002, 0x36b0000000000000 },
|
|
|
|
{ 0x00000001, 0x36a0000000000000 },
|
|
|
|
};
|
|
|
|
|
|
|
|
int sp_to_dp(long arg)
|
|
|
|
{
|
|
|
|
unsigned long dp;
|
|
|
|
|
|
|
|
__asm__("lfs 20,0(%0); stfd 20,0(%1)"
|
|
|
|
: : "b" (&sp_dp_equiv[arg].sp), "b" (&dp) : "memory");
|
|
|
|
if (dp != sp_dp_equiv[arg].dp) {
|
|
|
|
print_hex(sp_dp_equiv[arg].sp, 8);
|
|
|
|
print_string(" ");
|
|
|
|
print_hex(dp, 16);
|
|
|
|
print_string(" ");
|
|
|
|
print_hex(sp_dp_equiv[arg].dp, 16);
|
|
|
|
print_string(" ");
|
|
|
|
}
|
|
|
|
return dp != sp_dp_equiv[arg].dp;
|
|
|
|
}
|
|
|
|
|
|
|
|
int dp_to_sp(long arg)
|
|
|
|
{
|
|
|
|
unsigned int sp;
|
|
|
|
|
|
|
|
__asm__("lfd 21,0(%0); stfs 21,0(%1)"
|
|
|
|
: : "b" (&sp_dp_equiv[arg].dp), "b" (&sp) : "memory");
|
|
|
|
return sp != sp_dp_equiv[arg].sp;
|
|
|
|
}
|
|
|
|
|
|
|
|
int fpu_test_3(void)
|
|
|
|
{
|
|
|
|
int i, n, ret;
|
|
|
|
|
|
|
|
n = sizeof(sp_dp_equiv) / sizeof(sp_dp_equiv[0]);
|
|
|
|
enable_fp();
|
|
|
|
for (i = 0; i < n; ++i) {
|
|
|
|
ret = trapit(i, sp_to_dp);
|
|
|
|
if (ret != 0) {
|
|
|
|
if (ret == 1)
|
|
|
|
ret += i;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
ret = trapit(i, dp_to_sp);
|
|
|
|
if (ret != 0) {
|
|
|
|
if (ret == 1)
|
|
|
|
ret += i + 0x10000;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int fail = 0;
|
|
|
|
|
|
|
|
void do_test(int num, int (*test)(void))
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
print_test_number(num);
|
|
|
|
ret = test();
|
|
|
|
if (ret == 0) {
|
|
|
|
print_string("PASS\r\n");
|
|
|
|
} else {
|
|
|
|
fail = 1;
|
|
|
|
print_string("FAIL ");
|
|
|
|
print_hex(ret, 5);
|
|
|
|
print_string(" SRR0=");
|
|
|
|
print_hex(mfspr(SRR0), 16);
|
|
|
|
print_string(" SRR1=");
|
|
|
|
print_hex(mfspr(SRR1), 16);
|
|
|
|
print_string("\r\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
console_init();
|
|
|
|
|
|
|
|
do_test(1, fpu_test_1);
|
|
|
|
do_test(2, fpu_test_2);
|
|
|
|
do_test(3, fpu_test_3);
|
|
|
|
|
|
|
|
return fail;
|
|
|
|
}
|