#include <stddef.h> #include <stdint.h> #include <stdbool.h> #include "console.h" extern unsigned long callit(unsigned long arg1, unsigned long arg2, unsigned long (*fn)(unsigned long, unsigned long)); #define DSISR 18 #define DAR 19 #define SRR0 26 #define SRR1 27 #define PID 48 #define SPRG0 272 #define SPRG1 273 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)); } static inline void store_pte(unsigned long *p, unsigned long pte) { __asm__ volatile("stdbrx %1,0,%0" : : "r" (p), "r" (pte) : "memory"); } 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(':'); } #define DO_LARX(instr, addr, val) __asm__ volatile(instr " %0,0,%1" : "=r" (val) : "r" (addr)) #define DO_STCX(instr, addr, val, cc) __asm__ volatile(instr " %2,0,%1; mfcr %0" : "=r" (cc) \ : "r" (addr), "r" (val) : "cr0", "memory"); int resv_test_1(void) { unsigned long x, val, cc = 0; int count; x = 1234; for (count = 0; count < 1000; ++count) { DO_LARX("ldarx", &x, val); DO_STCX("stdcx.", &x, 5678, cc); if (cc & 0x20000000) break; } /* ldarx/stdcx. should succeed eventually */ if (count == 1000) return 1; if (x != 5678) return 2; for (count = 0; count < 1000; ++count) { DO_LARX("lwarx", &x, val); DO_STCX("stwcx.", &x, 9876, cc); if (cc & 0x20000000) break; } /* lwarx/stwcx. should succeed eventually */ if (count == 1000) return 3; if (x != 9876) return 4; for (count = 0; count < 1000; ++count) { DO_LARX("lharx", &x, val); DO_STCX("sthcx.", &x, 3210, cc); if (cc & 0x20000000) break; } /* lharx/sthcx. should succeed eventually */ if (count == 1000) return 5; if (x != 3210) return 6; return 0; } unsigned long do_larx(unsigned long size, unsigned long addr) { unsigned long val; switch (size) { case 1: DO_LARX("lbarx", addr, val); break; case 2: DO_LARX("lharx", addr, val); break; case 4: DO_LARX("lwarx", addr, val); break; case 8: DO_LARX("ldarx", addr, val); break; } return 0; } unsigned long do_stcx(unsigned long size, unsigned long addr) { unsigned long val = 0, cc; switch (size) { case 1: DO_STCX("stbcx.", addr, val, cc); break; case 2: DO_STCX("sthcx.", addr, val, cc); break; case 4: DO_STCX("stwcx.", addr, val, cc); break; case 8: DO_STCX("stdcx.", addr, val, cc); break; } return 0; } int resv_test_2(void) { unsigned long x[3]; unsigned long offset, j, size, ret; x[0] = 1234; x[1] = x[2] = 0; for (j = 0; j <= 3; ++j) { size = 1 << j; for (offset = 0; offset < 16; ++offset) { ret = callit(size, (unsigned long)&x[0] + offset, do_larx); if (ret == 0 && (offset & (size - 1)) != 0) return j + 1; if (ret == 0x600) { if ((offset & (size - 1)) == 0) return j + 0x10; } else if (ret) return ret; ret = callit(size, (unsigned long)&x[0] + offset, do_stcx); if (ret == 0 && (offset & (size - 1)) != 0) return j + 0x20; if (ret == 0x600) { if ((offset & (size - 1)) == 0) return j + 0x30; } else if (ret) 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, 4); print_string("\r\n"); } } int main(void) { console_init(); do_test(1, resv_test_1); do_test(2, resv_test_2); return fail; }