2012-10-12 22:25:22 +00:00
|
|
|
/*
|
|
|
|
TinyLoad - a simple region free (original) game launcher in 4k
|
|
|
|
|
|
|
|
# This code is licensed to you under the terms of the GNU GPL, version 2;
|
|
|
|
# see file COPYING or http://www.gnu.org/licenses/old-licenses/gpl-2.0.txt
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* This code comes from HBC's stub which was based on the Twilight Hack code */
|
|
|
|
// Copyright 2008-2009 Segher Boessenkool <segher@kernel.crashing.org>
|
|
|
|
// Copyright 2008-2009 Andre Heider <dhewg@wiibrew.org>
|
|
|
|
// Copyright 2008-2009 Hector Martin <marcan@marcansoft.com>
|
|
|
|
|
|
|
|
#include "debug.h"
|
|
|
|
#include "ios.h"
|
|
|
|
#include "cache.h"
|
|
|
|
#include "utils.h"
|
2012-10-13 16:57:03 +00:00
|
|
|
#include "di.h"
|
2012-10-12 22:25:22 +00:00
|
|
|
|
|
|
|
#define virt_to_phys(x) ((u32*)(((u32)(x))&0x3FFFFFFF))
|
|
|
|
#define phys_to_virt(x) ((u32*)(((u32)(x))|0x80000000))
|
|
|
|
|
|
|
|
// Low-level IPC access.
|
|
|
|
|
|
|
|
static inline u32 iread32(u32 addr)
|
|
|
|
{
|
|
|
|
u32 x;
|
|
|
|
|
|
|
|
asm volatile("lwz %0,0(%1) ; sync ; isync" : "=r"(x) : "b"(0xc0000000 | addr));
|
|
|
|
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void iwrite32(u32 addr, u32 x)
|
|
|
|
{
|
|
|
|
asm volatile("stw %0,0(%1) ; eieio" : : "r"(x), "b"(0xc0000000 | addr));
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef TINY
|
|
|
|
static u32 _ipc_read(u32 reg) __attribute__((noinline));
|
|
|
|
static void _ipc_write(u32 reg, u32 value) __attribute__((noinline));
|
|
|
|
static void ipc_bell(u32 w) __attribute__((noinline));
|
|
|
|
|
|
|
|
// inline the 4*, don't inline the 0x0d0 stuff. yes, this saves a few bytes.
|
|
|
|
static u32 _ipc_read(u32 reg) {
|
|
|
|
return iread32(0x0d000000 + reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _ipc_write(u32 reg, u32 value) {
|
|
|
|
iwrite32(0x0d000000 + reg, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline u32 ipc_read(u32 reg) {
|
|
|
|
return _ipc_read(4*reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void ipc_write(u32 reg, u32 value) {
|
|
|
|
_ipc_write(4*reg, value);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static u32 ipc_read(u32 reg) {
|
|
|
|
return iread32(0x0d000000 + 4*reg);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ipc_write(u32 reg, u32 value) {
|
|
|
|
iwrite32(0x0d000000 + 4*reg, value);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static void ipc_bell(u32 w)
|
|
|
|
{
|
|
|
|
ipc_write(1, w);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ios_delay(void) __attribute__((noinline));
|
|
|
|
static void ios_delay(void)
|
|
|
|
{
|
|
|
|
udelay(500);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ipc_wait_ack(void)
|
|
|
|
{
|
|
|
|
while (!(ipc_read(1) & 0x2)) {
|
|
|
|
//debug_string("no ack\n");
|
|
|
|
//udelay(10000);
|
|
|
|
}
|
|
|
|
ios_delay();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ipc_wait_reply(void)
|
|
|
|
{
|
|
|
|
while (!(ipc_read(1) & 0x4)) {
|
|
|
|
//debug_string("no reply\n");
|
|
|
|
//udelay(10000);
|
|
|
|
}
|
|
|
|
ios_delay();
|
|
|
|
}
|
|
|
|
|
|
|
|
static u32 ipc_wait(void)
|
|
|
|
{
|
|
|
|
u32 ret;
|
|
|
|
while (!((ret = ipc_read(1)) & 0x6)) {
|
|
|
|
//debug_string("no nothing\n");
|
|
|
|
//udelay(10000);
|
|
|
|
}
|
|
|
|
ios_delay();
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mid-level IPC access.
|
|
|
|
|
|
|
|
struct ipc {
|
|
|
|
u32 cmd;
|
|
|
|
int result;
|
|
|
|
int fd;
|
|
|
|
u32 arg[5];
|
|
|
|
|
|
|
|
u32 user[8];
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct ipc ipc ALIGNED(64);
|
|
|
|
|
|
|
|
void ipc_init(void)
|
|
|
|
{
|
|
|
|
ipc_write(1, 0x06);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ipc_send_request(void)
|
|
|
|
{
|
|
|
|
sync_after_write(&ipc, 0x40);
|
|
|
|
|
|
|
|
ipc_write(0, (u32)virt_to_phys(&ipc));
|
|
|
|
ipc_bell(1);
|
|
|
|
|
|
|
|
ipc_wait_ack();
|
|
|
|
|
|
|
|
//udelay(1000);
|
|
|
|
ipc_bell(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int ipc_send_twoack(void)
|
|
|
|
{
|
|
|
|
sync_after_write(&ipc, 0x40);
|
|
|
|
ios_delay();
|
|
|
|
|
|
|
|
ipc_write(0, (u32)virt_to_phys(&ipc));
|
|
|
|
ipc_bell(1);
|
|
|
|
|
|
|
|
if(ipc_wait() & 4) {
|
|
|
|
debug_string("got reply instead, bad\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
ipc_bell(2);
|
|
|
|
|
|
|
|
if(ipc_wait() & 4) {
|
|
|
|
debug_string("got reply instead, bad\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
ipc_bell(2);
|
|
|
|
ipc_bell(8);
|
|
|
|
debug_string("\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ipc_recv_reply(void)
|
|
|
|
{
|
|
|
|
for (;;) {
|
|
|
|
u32 reply;
|
|
|
|
|
|
|
|
ipc_wait_reply();
|
|
|
|
|
|
|
|
reply = ipc_read(2);
|
|
|
|
ipc_bell(4);
|
|
|
|
|
|
|
|
ipc_bell(8);
|
|
|
|
|
|
|
|
if (((u32*)reply) == virt_to_phys(&ipc))
|
|
|
|
break;
|
|
|
|
|
|
|
|
debug_string("Ignoring unexpected IPC reply @");
|
|
|
|
debug_uint((u32)reply);
|
|
|
|
debug_string("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
sync_before_read(&ipc, 0x40);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// High-level IPC access.
|
|
|
|
|
|
|
|
void ios_cleanup()
|
|
|
|
{
|
|
|
|
int loops = 0xA;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if((ipc_read(1) & 0x22) == 0x22)
|
|
|
|
ipc_write(1, (ipc_read(1)&~0x30) | 2);
|
|
|
|
if((ipc_read(1) & 0x14) == 0x14)
|
|
|
|
{
|
|
|
|
ipc_read(2);
|
|
|
|
ipc_write(1, (ipc_read(1)&~0x30) | 4);
|
|
|
|
ipc_write(12, 0x4000);
|
|
|
|
ipc_write(1, (ipc_read(1)&~0x30) | 8);
|
|
|
|
}
|
|
|
|
ipc_write(12, 0x4000);
|
|
|
|
udelay(1000);
|
|
|
|
loops--;
|
|
|
|
} while(loops != 0);
|
|
|
|
|
|
|
|
int fd;
|
|
|
|
for(fd = 0; fd != 31; fd++)
|
|
|
|
ios_close(fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
int ios_open(const char *filename, u32 mode)
|
|
|
|
{
|
|
|
|
#ifdef TINY
|
|
|
|
sync_after_write((void*)filename, 0x20);
|
|
|
|
#else
|
|
|
|
sync_after_write((void*)filename, strlen(filename) + 1);
|
|
|
|
#endif
|
|
|
|
#ifndef TINY
|
|
|
|
memset(&ipc, 0, sizeof ipc);
|
|
|
|
#endif
|
|
|
|
ipc.cmd = 1;
|
|
|
|
ipc.fd = 0;
|
|
|
|
ipc.arg[0] = (u32)virt_to_phys(filename);
|
|
|
|
ipc.arg[1] = mode;
|
|
|
|
|
|
|
|
//debug_string("Sending openreq\n");
|
|
|
|
ipc_send_request();
|
|
|
|
//debug_string("AAA\n");
|
|
|
|
ipc_recv_reply();
|
|
|
|
//debug_string("BBB\n");
|
|
|
|
|
|
|
|
return ipc.result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ios_std(int fd, int cmd)
|
|
|
|
{
|
|
|
|
ipc.cmd = cmd;
|
|
|
|
ipc.fd = fd;
|
|
|
|
|
|
|
|
ipc_send_request();
|
|
|
|
ipc_recv_reply();
|
|
|
|
}
|
|
|
|
|
|
|
|
int ios_close(int fd)
|
|
|
|
{
|
|
|
|
#ifndef TINY
|
|
|
|
memset(&ipc, 0, sizeof ipc);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
ios_std(fd, 2);
|
|
|
|
|
|
|
|
return ipc.result;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ios_read(int fd, void *buf, u32 size)
|
|
|
|
{
|
|
|
|
#ifndef TINY
|
|
|
|
memset(&ipc, 0, sizeof ipc);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
ipc.arg[0] = (u32)virt_to_phys(buf);
|
|
|
|
ipc.arg[1] = size;
|
|
|
|
|
|
|
|
ios_std(fd, 3);
|
|
|
|
|
|
|
|
sync_before_read(buf, size);
|
|
|
|
|
|
|
|
return ipc.result;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ios_ioctl(int fd, u32 n, void *in, u32 in_size, void *out, u32 out_size)
|
|
|
|
{
|
|
|
|
#ifndef TINY
|
|
|
|
memset(&ipc, 0, sizeof ipc);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if(in)
|
|
|
|
sync_after_write(in, in_size);
|
|
|
|
if(out)
|
|
|
|
sync_after_write(out, out_size);
|
|
|
|
|
|
|
|
ipc.arg[0] = n;
|
|
|
|
ipc.arg[1] = (u32)virt_to_phys(in);
|
|
|
|
ipc.arg[2] = in_size;
|
|
|
|
ipc.arg[3] = (u32)virt_to_phys(out);
|
|
|
|
ipc.arg[4] = out_size;
|
|
|
|
|
|
|
|
ios_std(fd, 6);
|
|
|
|
|
|
|
|
if(out)
|
|
|
|
sync_before_read(out, out_size);
|
|
|
|
|
|
|
|
return ipc.result;
|
|
|
|
}
|
|
|
|
|
|
|
|
int _ios_ioctlv(int fd, u32 n, u32 in_count, u32 out_count, struct ioctlv *vec, int reboot)
|
|
|
|
{
|
|
|
|
u32 i;
|
|
|
|
|
|
|
|
#ifndef TINY
|
|
|
|
memset(&ipc, 0, sizeof ipc);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
for (i = 0; i < in_count + out_count; i++) {
|
|
|
|
if (vec[i].data) {
|
|
|
|
sync_after_write(vec[i].data, vec[i].len);
|
|
|
|
vec[i].data = (void *)virt_to_phys(vec[i].data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sync_after_write(vec, (in_count + out_count) * sizeof *vec);
|
|
|
|
|
|
|
|
ipc.cmd = 7;
|
|
|
|
ipc.fd = fd;
|
|
|
|
ipc.arg[0] = n;
|
|
|
|
ipc.arg[1] = in_count;
|
|
|
|
ipc.arg[2] = out_count;
|
|
|
|
ipc.arg[3] = (u32)virt_to_phys(vec);
|
|
|
|
|
|
|
|
if(reboot) {
|
|
|
|
//debug_string("Sending twoack\n");
|
|
|
|
if(ipc_send_twoack())
|
|
|
|
return 0;
|
|
|
|
debug_string("Reboot returned a reply instead of an ACK");
|
|
|
|
} else {
|
|
|
|
//debug_string("Sending request\n");
|
|
|
|
ipc_send_request();
|
|
|
|
//debug_string("K\n");
|
|
|
|
}
|
|
|
|
ipc_recv_reply();
|
|
|
|
//debug_string("Got reply\n");
|
|
|
|
|
|
|
|
for (i = in_count; i < in_count + out_count; i++) {
|
|
|
|
if (vec[i].data) {
|
|
|
|
vec[i].data = phys_to_virt((u32)vec[i].data);
|
|
|
|
sync_before_read(vec[i].data, vec[i].len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(reboot && (ipc.result >= 0))
|
|
|
|
return -100;
|
|
|
|
return ipc.result;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ios_ioctlv(int fd, u32 n, u32 in_count, u32 out_count, struct ioctlv *vec) {
|
|
|
|
return _ios_ioctlv(fd, n, in_count, out_count, vec, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int ios_ioctlvreboot(int fd, u32 n, u32 in_count, u32 out_count, struct ioctlv *vec) {
|
|
|
|
return _ios_ioctlv(fd, n, in_count, out_count, vec, 1);
|
|
|
|
}
|