/* $NetBSD: disptest.c,v 1.6 2009/03/18 10:22:29 cegger Exp $ */
/*-
* Copyright (c) 1999 Shin Takemura.
* All rights reserved.
*
* This software is part of the PocketBSD.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
* This product includes software developed by the PocketBSD project
* and its contributors.
* 4. Neither the name of the project nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*
*/
#include <pbsdboot.h>
extern BOOL SetKMode(BOOL);
#define ARRAYSIZEOF(a) (sizeof(a)/sizeof(*(a)))
static struct area {
long start, end;
} targets[] = {
{ 0x0a000000, 0x0b000000 },
{ 0x10000000, 0x14000000 },
};
int ntargets = ARRAYSIZEOF(targets);
void
flush_XX(void)
{
static volatile unsigned char tmp[1024*64];
int i, s;
for (i = 0; i < ARRAYSIZEOF(tmp); i++) {
s += tmp[i];
}
}
static void
gpio_test(void)
{
#define GIUBASE 0xab000000
#define GIUOFFSET 0x0100
volatile unsigned short *giusell;
volatile unsigned short *giuselh;
volatile unsigned short *giupiodl;
volatile unsigned short *giupiodh;
unsigned short sell = 0;
unsigned short selh = 0;
unsigned short piodl = 0;
unsigned short piodh = 0;
int res, i;
unsigned short regs[16];
unsigned short prev_regs[16];
unsigned char* p = (char*)VirtualAlloc(0, 1024, MEM_RESERVE,
PAGE_NOACCESS);
res = VirtualCopy((LPVOID)p, (LPVOID)(GIUBASE >> 8), 1024,
PAGE_READWRITE|PAGE_NOCACHE|PAGE_PHYSICAL);
if (!res) {
win_printf(TEXT("VirtualCopy() failed."));
}
for (i = 0; i < 16; i++) {
prev_regs[i] = ~0;
}
giusell = (unsigned short*)(p + GIUOFFSET + 0);
giuselh = (unsigned short*)(p + GIUOFFSET + 2);
giupiodl = (unsigned short*)(p + GIUOFFSET + 4);
giupiodh = (unsigned short*)(p + GIUOFFSET + 6);
while (1) {
sell = *giusell;
selh = *giuselh;
*giusell = sell;
*giuselh = selh;
piodl = *giupiodl;
piodh = *giupiodh;
*giupiodl = piodl;
*giupiodh = piodh;
for (i = 0; i < 16; i++) {
regs[i] = *(unsigned short*)(p + GIUOFFSET + i * 2);
}
for (i = 0; i < 16; i++) {
if (i != 3 && i != 5 && regs[i] != prev_regs[i]) {
win_printf(TEXT("IOSEL=%04x%04x "),
regs[1], regs[0]);
win_printf(TEXT("PIOD=%04x%04x "),
regs[3], regs[2]);
win_printf(TEXT("STAT=%04x%04x "),
regs[5], regs[4]);
win_printf(TEXT("(%04x%04x) "),
regs[5]®s[7], regs[4]®s[6]);
win_printf(TEXT("EN=%04x%04x "),
regs[7], regs[6]);
win_printf(TEXT("TYP=%04x%04x "),
regs[9], regs[8]);
win_printf(TEXT("ALSEL=%04x%04x "),
regs[11], regs[10]);
win_printf(TEXT("HTSEL=%04x%04x "),
regs[13], regs[12]);
win_printf(TEXT("PODAT=%04x%04x "),
regs[15], regs[14]);
win_printf(TEXT("\n"));
for (i = 0; i < 16; i++) {
prev_regs[i] = regs[i];
}
break;
}
}
}
}
static struct regdesc {
TCHAR *name;
int physaddr;
int size;
int mask;
//void *addr;
unsigned long val;
unsigned long preval;
} test_regs[] = {
#if 0
/*
* Vrc4172 GPIO and PWM
*/
{ TEXT("EXGPDATA0"), 0x15001080, 2, 0xfffd },
{ TEXT("EXGPDATA1"), 0x150010c0, 2, 0xffff },
{ TEXT("LCDDUTYEN"), 0x15003880, 2, 0xffff },
{ TEXT("LCDFREQ"), 0x15003882, 2, 0xffff },
{ TEXT("LCDDUTY"), 0x15003884, 2, 0xffff },
#endif
#if 0
/*
* Vr41xx GPIO
*/
{ TEXT("GIUPIODL"), 0x0b000104, 2, 0xffff },
{ TEXT("GIUPIODH"), 0x0b000106, 2, 0xffff },
{ TEXT("GIUPODATL"), 0x0b00011c, 2, 0xffff },
{ TEXT("GIUPODATH"), 0x0b00011e, 2, 0xffff },
{ TEXT("GIUUSEUPDN"), 0x0b0002e0, 2, 0xffff },
{ TEXT("GIUTERMUPDN"), 0x0b0002e2, 2, 0xffff },
#endif
/*
* MQ200
*/
{ TEXT("PM00R"), 0x0a600000, 4, 0xffffffff },
{ TEXT("PM01R"), 0x0a600004, 4, 0xffffffff },
{ TEXT("PM02R"), 0x0a600008, 4, 0xffffffff },
{ TEXT("PM06R"), 0x0a600018, 4, 0xffffffff },
{ TEXT("PM07R"), 0x0a60001c, 4, 0xffffffff },
{ TEXT("CC00R"), 0x0a602000, 4, 0x0000003f },
{ TEXT("CC01R"), 0x0a602004, 4, 0x00000000 },
{ TEXT("MM00R"), 0x0a604000, 4, 0x00000007 },
{ TEXT("MM01R"), 0x0a604004, 4, 0xffffffff },
{ TEXT("MM02R"), 0x0a604008, 4, 0xffffffff },
{ TEXT("MM03R"), 0x0a60400c, 4, 0x00000001 },
{ TEXT("MM04R"), 0x0a604010, 4, 0x00000001 },
{ TEXT("IN00R"), 0x0a608000, 4, 0x0000001f },
{ TEXT("IN01R"), 0x0a608004, 4, 0x0000ffff },
{ TEXT("IN02R"), 0x0a608008, 4, 0x00000000 },
{ TEXT("IN03R"), 0x0a60800c, 4, 0x00000000 },
{ TEXT("GC00R"), 0x0a60a000, 4, 0xfffff9ff },
{ TEXT("GC01R"), 0x0a60a004, 4, 0x10ffffff },
{ TEXT("GC20R"), 0x0a60a080, 4, 0xffffffff },
{ TEXT("GC21R"), 0x0a60a084, 4, 0x0000007f },
{ TEXT("FP00R"), 0x0a60e000, 4, 0xffffffff },
{ TEXT("FP01R"), 0x0a60e004, 4, 0xffffffff },
{ TEXT("FP02R"), 0x0a60e008, 4, 0x007fffff },
{ TEXT("FP03R"), 0x0a60e00c, 4, 0x0707003f },
{ TEXT("FP04R"), 0x0a60e010, 4, 0xffff3fff },
{ TEXT("FP05R"), 0x0a60e014, 4, 0xffffffff },
{ TEXT("FP0FR"), 0x0a60e03c, 4, 0xffffffff },
{ TEXT("DC00R"), 0x0a614000, 4, 0xffffffff },
{ TEXT("DC01R"), 0x0a614004, 4, 0x0000003f },
{ TEXT("DC02R"), 0x0a614008, 4, 0xffffffff },
{ TEXT("DC03R"), 0x0a61400c, 4, 0xffffffff },
{ TEXT("PC00R"), 0x0a616000, 4, 0xffffffff },
{ TEXT("PC04R"), 0x0a616004, 4, 0xffffffff },
{ TEXT("PC08R"), 0x0a616008, 4, 0xffffffff },
{ TEXT("PC0CR"), 0x0a61600c, 4, 0xffffffff },
{ TEXT("PC10R"), 0x0a616010, 4, 0xffffffff },
{ TEXT("PC14R"), 0x0a616014, 4, 0xffffffff },
{ TEXT("PC2CR"), 0x0a61602c, 4, 0xffffffff },
{ TEXT("PC3CR"), 0x0a61603c, 4, 0xffffffff },
{ TEXT("PC40R"), 0x0a616040, 4, 0xffffffff },
{ TEXT("PC44R"), 0x0a616044, 4, 0x00000003 },
};
extern int SetKMode(int);
static void
regfetch(struct regdesc* desc)
{
SetKMode(1);
switch (desc->size) {
case 1:
desc->val = *(unsigned char*)(desc->physaddr | 0xa0000000);
break;
case 2:
desc->val = *(unsigned short*)(desc->physaddr | 0xa0000000);
break;
case 4:
desc->val = *(unsigned long*)(desc->physaddr | 0xa0000000);
break;
default:
win_printf(TEXT("Invalid size"));
break;
}
SetKMode(0);
desc->val &= desc->mask;
}
static void
register_test(void)
{
int i;
int nregs = sizeof(test_regs)/sizeof(*test_regs);
for (i = 0; i < nregs; i++) {
regfetch(&test_regs[i]);
test_regs[i].preval = test_regs[i].val;
}
while (1) {
for (i = 0; i < nregs; i++) {
regfetch(&test_regs[i]);
if (test_regs[i].val != test_regs[i].preval) {
win_printf(TEXT("%20s(%08x) %08x -> %08x\n"),
test_regs[i].name,
test_regs[i].physaddr,
test_regs[i].preval,
test_regs[i].val);
test_regs[i].preval = test_regs[i].val;
}
}
Sleep(10); /* 10 msec */
}
}
static void
dump_memory(void)
{
HANDLE fh = INVALID_HANDLE_VALUE;
#define UNICODE_MEMORY_CARD \
TEXT('\\'), 0xff92, 0xff93, 0xff98, TEXT(' '), 0xff76, 0xff70, \
0xff84, 0xff9e
TCHAR filename[] = { UNICODE_MEMORY_CARD, TEXT('2'), TEXT('\\'),
TEXT('d'), TEXT('u'), TEXT('m'), TEXT('p'), 0 };
unsigned long *addr;
int found;
win_printf(TEXT("dump to %s\n"), filename);
fh = CreateFile(
filename, /* file name */
GENERIC_WRITE, /* access (read-write) mode */
FILE_SHARE_WRITE,/* share mode */
NULL, /* pointer to security attributes */
CREATE_ALWAYS, /* how to create */
FILE_ATTRIBUTE_NORMAL, /* file attributes*/
NULL /* handle to file with attributes to */
);
if (fh == INVALID_HANDLE_VALUE) {
return;
}
for (addr = (unsigned long*)0xbe000000;
addr < (unsigned long*)0xbfffffff;
addr += 2048) {
char buf[2048];
DWORD n;
SetKMode(1);
memcpy(buf, addr, 2048);
SetKMode(0);
if (WriteFile(fh, buf, 2048, &n, NULL) == 0 ||
n != 2048) {
win_printf(TEXT("dump failed\n"));
break;
}
}
CloseHandle(fh);
}
static void
serial_test(void)
{
#if 1
# define SIUADDR 0xac000000
# define REGOFFSET 0x0
#else
# define SIUADDR 0xab000000
# define REGOFFSET 0x1a0
#endif
#define REGSIZE 32
int i, changed, res;
unsigned char regs[REGSIZE], prev_regs[REGSIZE];
unsigned char* p = (char*)VirtualAlloc(0, 1024, MEM_RESERVE,
PAGE_NOACCESS);
for (i = 0; i < ARRAYSIZEOF(prev_regs); i++) {
prev_regs[i] = ~0;
}
res = VirtualCopy((LPVOID)p, (LPVOID)(SIUADDR >> 8), 1024,
PAGE_READWRITE|PAGE_NOCACHE|PAGE_PHYSICAL);
if (!res) {
win_printf(TEXT("VirtualCopy() failed."));
}
while (1) {
flush_XX();
for (i = 0; i < ARRAYSIZEOF(regs); i++) {
regs[i] = p[REGOFFSET + i];
}
changed = 0;
for (i = 0; i < ARRAYSIZEOF(regs); i++) {
if (regs[i] != prev_regs[i]) {
changed++;
}
prev_regs[i] = regs[i];
}
if (changed) {
win_printf(TEXT("SIU regs: "));
for (i = 0; i < ARRAYSIZEOF(regs); i++) {
win_printf(TEXT("%02x "), regs[i]);
}
win_printf(TEXT("\n"));
}
}
VirtualFree(p, 0, MEM_RELEASE);
}
static long
checksum(char* addr, int size)
{
long sum = 0;
int i;
for (i = 0; i < size; i++) {
sum += *addr++ * i;
}
return (sum);
}
static int
examine(char* addr, int size)
{
long random_data[256];
long dijest;
int i;
for (i = 0; i < ARRAYSIZEOF(random_data); i++) {
random_data[i] = Random();
}
if (sizeof(random_data) < size) {
size = sizeof(random_data);
}
memcpy(addr, (char*)random_data, size);
dijest= checksum((char*)random_data, size);
return (dijest == checksum(addr, size));
}
void
display_search(void)
{
int step = 0x10000;
int i;
long addr;
for (i = 0; i < ntargets; i++) {
int prevres = -1;
for (addr = targets[i].start;
addr < targets[i].end;
addr += step) {
int res;
#if 0
char* p = (char*)VirtualAlloc(0, step, MEM_RESERVE,
PAGE_NOACCESS);
res = VirtualCopy((LPVOID)p, (LPVOID)(addr >> 8), step,
PAGE_READWRITE|PAGE_NOCACHE|PAGE_PHYSICAL);
if (!res) {
win_printf(TEXT("VirtualCopy() failed."));
}
res = examine(p, step);
VirtualFree(p, 0, MEM_RELEASE);
#else
SetKMode(1);
res = examine((char*)((int)addr | 0xa0000000), step);
SetKMode(0);
#endif
if (res != prevres && prevres != -1) {
if (res) {
win_printf(TEXT("0x%x "), addr);
} else {
win_printf(TEXT("- 0x%x\n"), addr);
}
} else
if (res && prevres == -1) {
win_printf(TEXT("0x%x "), addr);
}
prevres = res;
}
if (prevres) {
win_printf(TEXT("\n"));
}
}
}
void
display_draw(void)
{
long addr = 0x13000000;
int size = 0x80000;
char* p;
int i, j, res;
int x, y;
int stride = 1280;
p = (char*)VirtualAlloc(0, size, MEM_RESERVE,
PAGE_NOACCESS);
res = VirtualCopy((LPVOID)p, (LPVOID)(addr >> 8), size,
PAGE_READWRITE|PAGE_NOCACHE|PAGE_PHYSICAL);
if (!res) {
win_printf(TEXT("VirtualCopy() failed."));
}
for (i = 0; i < 10000; i++) {
p[i] = i;
}
for (x = 0; x < 640; x += 10) {
for (y = 0; y < 240; y += 1) {
p[stride * y + x] = (char)0xff;
}
}
for (y = 0; y < 240; y += 10) {
for (x = 0; x < 640; x += 1) {
p[stride * y + x] = (char)0xff;
}
}
for (i = 0; i < 16; i++) {
for (j = 0; j < 16; j++) {
for (x = i * 32; x < i * 32 + 32; x++) {
for (y = j * 15; y < j * 15 + 15; y++) {
p[stride * y + x] = j * 16 + i;
}
}
}
}
VirtualFree(p, 0, MEM_RELEASE);
}
#define PCIC_IDENT 0x00
#define PCIC_REG_INDEX 0
#define PCIC_REG_DATA 1
#define PCIC_IDENT_EXPECTED 0x83
void
pcic_search(void)
{
long addr;
int window_size = 0x10000;
int i;
for (addr = 0x14000000; addr < 0x18000000; addr += window_size) {
int res;
unsigned char* p;
p = (char*)VirtualAlloc(0, window_size, MEM_RESERVE,
PAGE_NOACCESS);
res = VirtualCopy((LPVOID)p, (LPVOID)(addr >> 8), window_size,
PAGE_READWRITE|PAGE_NOCACHE|PAGE_PHYSICAL);
if (!res) {
win_printf(TEXT("VirtualCopy() failed."));
}
for (i = 0; i < window_size; i += 2) {
p[i + PCIC_REG_INDEX] = PCIC_IDENT;
if (p[i + PCIC_REG_DATA] == PCIC_IDENT_EXPECTED) {
win_printf(TEXT("pcic is found at 0x%x\n"),
addr + i);
}
}
VirtualFree(p, 0, MEM_RELEASE);
}
}
#define VRPCIU_CONFA (*(u_int32_t*)0xaf000c18)
#define VRPCIU_CONFD (*(u_int32_t*)0xaf000c14)
void
pci_dump(void)
{
int mode, i;
BOOL SetKMode(BOOL);
int bus, dev;
u_int32_t addr, val;
u_int32_t addrs[] = {
0x00000800,
0x00001000,
0x00002000,
0x00004000,
0x00008000,
0x00010000,
0x00020000,
0x00040000,
0x00080000,
0x00100000,
0x00200000,
0x00400000,
0x00800000,
0x01000000,
0x02000000,
0x04000000,
0x08000000,
0x10000000,
0x20000000,
0x40000000,
0x80000000,
};
#if 0 /* You can find Vrc4173 BCU at 0xb6010000 on Sigmarion II */
win_printf(TEXT("Vrc4173 CMUCLKMSK: %04X\n"),
*(u_int16_t*)0xb6010040);
win_printf(TEXT("Vrc4173 CMUSRST: %04X\n"),
*(u_int16_t*)0xb6010042);
/* enable CARDU clock */
*(u_int16_t*)0xb6010042 = 0x0006; /* enable CARD1RST and CARD2RST */
*(u_int16_t*)0xb6010040 = *(u_int16_t*)0xb6010040 | 0x00c0;
*(u_int16_t*)0xb6010042 = 0x0000; /* disable CARD1RST and CARD2RST */
win_printf(TEXT("Vrc4173 CMUCLKMSK: %04X\n"),
*(u_int16_t*)0xb6010040);
win_printf(TEXT("Vrc4173 CMUSRST: %04X\n"),
*(u_int16_t*)0xb6010042);
#endif
for (i = 0; i < sizeof(addrs)/sizeof(*addrs); i++) {
VRPCIU_CONFA = addrs[i];
val = VRPCIU_CONFD;
win_printf(TEXT("%2d: %08X %04X %04X\n"),
i, addrs[i], val & 0xffff, (val >> 16) & 0xffff);
}
mode = SetKMode(1);
SetKMode(mode);
}
void
hardware_test(void)
{
int do_gpio_test = 0;
int do_register_test = 0;
int do_serial_test = 0;
int do_display_draw = 0;
int do_display_search = 0;
int do_pcic_search = 0;
int do_dump_memory = 0;
int do_pci_dump = 0;
if (do_gpio_test) {
gpio_test();
}
if (do_register_test) {
register_test();
}
if (do_serial_test) {
serial_test();
}
if (do_display_draw) {
display_draw();
}
if (do_display_search) {
display_search();
}
if (do_pcic_search) {
pcic_search();
}
if (do_dump_memory) {
dump_memory();
}
if (do_pci_dump) {
pci_dump();
}
}