397 lines
9.8 KiB
C
Raw Normal View History

2012-01-21 20:57:41 +00:00
/*-------------------------------------------------------------
usbstorage_starlet.c -- USB mass storage support, inside starlet
Copyright (C) 2011 Dimok
Copyright (C) 2011 Rodries
Copyright (C) 2009 Kwiirk
2012-01-21 20:57:41 +00:00
If this driver is linked before libogc, this will replace the original
usbstorage driver by svpe from libogc
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any
damages arising from the use of this software.
2012-01-21 20:57:41 +00:00
Permission is granted to anyone to use this software for any
purpose, including commercial applications, and to alter it and
redistribute it freely, subject to the following restrictions:
2012-01-21 20:57:41 +00:00
1. The origin of this software must not be misrepresented; you
must not claim that you wrote the original software. If you use
this software in a product, an acknowledgment in the product
documentation would be appreciated but is not required.
2012-01-21 20:57:41 +00:00
2. Altered source versions must be plainly marked as such, and
must not be misrepresented as being the original software.
2012-01-21 20:57:41 +00:00
3. This notice may not be removed or altered from any source
distribution.
2012-01-21 20:57:41 +00:00
-------------------------------------------------------------*/
2012-01-21 20:57:41 +00:00
#include <gccore.h>
#include <malloc.h>
#include <stdio.h>
#include <string.h>
#include "memory/mem2.hpp"
#include "usbstorage.h"
#include "usbstorage_libogc.h"
#include "usbthread.h"
#include "gecko/gecko.hpp"
2012-01-21 20:57:41 +00:00
/* IOCTL commands */
#define UMS_BASE (('U'<<24)|('M'<<16)|('S'<<8))
#define USB_IOCTL_UMS_INIT (UMS_BASE+0x1)
#define USB_IOCTL_UMS_GET_CAPACITY (UMS_BASE+0x2)
#define USB_IOCTL_UMS_READ_SECTORS (UMS_BASE+0x3)
#define USB_IOCTL_UMS_WRITE_SECTORS (UMS_BASE+0x4)
#define USB_IOCTL_UMS_READ_STRESS (UMS_BASE+0x5)
#define USB_IOCTL_UMS_SET_VERBOSE (UMS_BASE+0x6)
#define USB_IOCTL_UMS_UMOUNT (UMS_BASE+0x10)
#define USB_IOCTL_UMS_WATCHDOG (UMS_BASE+0x80)
#define USB_IOCTL_UMS_TESTMODE (UMS_BASE+0x81)
#define USB_IOCTL_SET_PORT (UMS_BASE+0x83)
#define MAX_SECTOR_SIZE 4096
#define MAX_BUFFER_SECTORS 128
#define UMS_HEAPSIZE 2*1024
#define USB_MEM2_SIZE MAX_BUFFER_SECTORS*MAX_SECTOR_SIZE
2012-01-21 20:57:41 +00:00
/* Variables */
static char fs[] ATTRIBUTE_ALIGN(32) = "/dev/usb2";
static char fs2[] ATTRIBUTE_ALIGN(32) = "/dev/usb123";
static char fs3[] ATTRIBUTE_ALIGN(32) = "/dev/usb/ehc";
u8 *mem2_ptr = NULL;
s32 hid = -1, fd = -1;
s8 usb2_port = -1; //current USB port
bool hddInUse[2] = { false, false };
u32 hdd_sector_size[2] = { 512, 512 };
bool first = false;
int usb_libogc_mode = 0;
2012-01-21 20:57:41 +00:00
static inline s32 __USBStorage_isMEM2Buffer(const void *buffer)
{
u32 high_addr = ((u32)buffer) >> 24;
return(high_addr == 0x90) || (high_addr == 0xD0);
}
s32 USBStorage2_Init(u32 port)
2012-01-21 20:57:41 +00:00
{
/* allocate buf2 */
if(mem2_ptr == NULL)
mem2_ptr = MEM2_alloc(USB_MEM2_SIZE);
2012-01-21 20:57:41 +00:00
if(usb_libogc_mode)
{
USBKeepAliveThreadReset();
__io_usbstorage_ogc.startup();
return (USBStorage2_GetCapacity(port, &hdd_sector_size[port]) == 0) ? IPC_ENOENT : 0;
}
if(hddInUse[port])
return 0;
/* Create heap */
if(hid < 0)
2012-01-21 20:57:41 +00:00
{
hid = iosCreateHeap(UMS_HEAPSIZE);
if (hid < 0) return IPC_ENOMEM;
2012-01-21 20:57:41 +00:00
}
/* Open USB device */
if (fd < 0) fd = IOS_Open(fs, 0);
if (fd < 0) fd = IOS_Open(fs2, 0);
if (fd < 0) fd = IOS_Open(fs3, 0);
if (fd < 0) return fd;
USBStorage2_SetPort(port);
2012-01-21 20:57:41 +00:00
/* Initialize USB storage */
IOS_IoctlvFormat(hid, fd, USB_IOCTL_UMS_INIT, ":");
2012-01-21 20:57:41 +00:00
/* Get device capacity */
if(USBStorage2_GetCapacity(port, &hdd_sector_size[port]) == 0)
return IPC_ENOENT;
hddInUse[port] = true;
2012-01-21 20:57:41 +00:00
return 0; // 0->HDD, 1->DVD
}
2012-01-21 20:57:41 +00:00
void USBStorage2_Deinit()
{
/* Close USB device */
if(usb_libogc_mode)
{
USBKeepAliveThreadReset();
__io_usbstorage_ogc.shutdown();
}
else if(fd >= 0)
IOS_Close(fd); // not sure to close the fd is needed
if(mem2_ptr != NULL)
MEM2_free(mem2_ptr);
mem2_ptr = NULL;
/* Reset Variables */
if(usb2_port == 0 || usb2_port == 1)
2012-01-21 20:57:41 +00:00
{
hddInUse[usb2_port] = false;
usb2_port = -1;
2012-01-21 20:57:41 +00:00
}
fd = -1;
2012-01-21 20:57:41 +00:00
}
s32 USBStorage2_SetPort(s8 port)
{
//! Port = 2 is handle in the loader, no need to handle it in cIOS
if(port > 1 || port < 0)
return -1;
if(port == usb2_port)
return 0;
s32 ret = -1;
usb2_port = port;
first = true;
gprintf("Changing USB port to port %i....\n", port);
//must be called before USBStorage2_Init (default port 0)
if(fd >= 0)
ret = IOS_IoctlvFormat(hid, fd, USB_IOCTL_SET_PORT, "i:", usb2_port);
return ret;
}
s8 USBStorage2_GetPort()
{
return usb2_port;
}
s32 USBStorage2_GetCapacity(u32 port, u32 *_sector_size)
2012-01-21 20:57:41 +00:00
{
if((usb_libogc_mode && !__io_usbstorage_ogc.isInserted()) || (!usb_libogc_mode && fd < 0))
return 0;
u32 numSectors = 0;
u32 sectorSize = 0;
USBStorage2_SetPort(port);
if(usb_libogc_mode)
{
USBKeepAliveThreadReset();
USB_OGC_GetCapacity(&numSectors, &sectorSize);
}
else
numSectors = IOS_IoctlvFormat(hid, fd, USB_IOCTL_UMS_GET_CAPACITY, ":i", &sectorSize);
if(first && numSectors && sectorSize)
{
gprintf(" * * * * * * * * * * * *\n");
gprintf(" * HDD Information\n * Sectors: %lu\n", numSectors);
u32 size = ((((numSectors / 1024U) * sectorSize) / 1024U) / 1024U);
if(size >= 1000U)
gprintf(" * Size [Sector Size]: %lu.%lu TB [%u]\n", size / 1024U, (size * 100U) % 1024U, sectorSize);
else
gprintf(" * Size [Sector Size]: %lu GB [%u]\n", size, sectorSize);
gprintf(" * * * * * * * * * * * *\n");
first = false;
2012-01-21 20:57:41 +00:00
}
if(numSectors && _sector_size)
*_sector_size = sectorSize;
return numSectors;
2012-01-21 20:57:41 +00:00
}
s32 USBStorage2_ReadSectors(u32 port, u32 sector, u32 numSectors, void *buffer)
2012-01-21 20:57:41 +00:00
{
s32 ret = -1;
if(usb_libogc_mode)
{
USBKeepAliveThreadReset();
reading = true;
ret = __io_usbstorage_ogc.readSectors(sector, numSectors, buffer);
reading = false;
return ret;
}
bool isMEM2Buffer = __USBStorage_isMEM2Buffer(buffer);
u8 *buf = (u8 *)buffer;
2012-01-21 20:57:41 +00:00
/* Device not opened */
if(fd < 0)
return fd;
2012-01-21 20:57:41 +00:00
/* MEM1 buffer */
if(!isMEM2Buffer)
{
buf = mem2_ptr;
if(!buf)
return IPC_ENOMEM;
}
USBStorage2_SetPort(port);
2012-01-21 20:57:41 +00:00
s32 read_secs, read_size;
2012-01-21 20:57:41 +00:00
while(numSectors > 0)
2012-01-21 20:57:41 +00:00
{
read_secs = numSectors > MAX_BUFFER_SECTORS ? MAX_BUFFER_SECTORS : numSectors;
read_size = read_secs * hdd_sector_size[port];
/* Read data */
ret = IOS_IoctlvFormat(hid, fd, USB_IOCTL_UMS_READ_SECTORS, "ii:d", sector, read_secs, buf, read_size);
if(ret < 0)
return ret;
/* Copy data */
if(!isMEM2Buffer)
{
memcpy(buffer, buf, read_size);
buffer += read_size;
}
else
buf += read_size;
sector += read_secs;
numSectors -= read_secs;
2012-01-21 20:57:41 +00:00
}
return ret;
}
s32 USBStorage2_WriteSectors(u32 port, u32 sector, u32 numSectors, const void *buffer)
2012-01-21 20:57:41 +00:00
{
if(usb_libogc_mode)
{
USBKeepAliveThreadReset();
return __io_usbstorage_ogc.writeSectors(sector, numSectors, buffer);
}
bool isMEM2Buffer = __USBStorage_isMEM2Buffer(buffer);
u8 *buf = (u8 *)buffer;
s32 ret = -1;
2012-01-21 20:57:41 +00:00
/* Device not opened */
if(fd < 0)
return fd;
USBStorage2_SetPort(port);
s32 write_size, write_secs;
while(numSectors > 0)
2012-01-21 20:57:41 +00:00
{
write_secs = numSectors > MAX_BUFFER_SECTORS ? MAX_BUFFER_SECTORS : numSectors;
write_size = write_secs * hdd_sector_size[port];
2012-01-21 20:57:41 +00:00
/* MEM1 buffer */
if(!isMEM2Buffer)
{
// Do not read more than MAX_BUFFER_SECTORS sectors at once and create a mem overflow!
memcpy(mem2_ptr, buf, write_size);
2012-01-21 20:57:41 +00:00
ret = IOS_IoctlvFormat(hid, fd, USB_IOCTL_UMS_WRITE_SECTORS, "ii:d", sector, write_secs, mem2_ptr, write_size);
if(ret < 0)
return ret;
}
else
{
/* Write data */
ret = IOS_IoctlvFormat(hid, fd, USB_IOCTL_UMS_WRITE_SECTORS, "ii:d", sector, write_secs, buf, write_size);
if(ret < 0)
return ret;
}
2012-01-21 20:57:41 +00:00
sector += write_secs;
numSectors -= write_secs;
buf += write_size;
}
2012-01-21 20:57:41 +00:00
return ret;
}
s32 USBStorage2_GetSectorSize()
{
if(usb2_port == 0 || usb2_port == 1)
return hdd_sector_size[usb2_port];
return 0;
}
static bool __usbstorage_Startup(void)
{
return USBStorage2_Init(0) >= 0;
}
2012-01-21 20:57:41 +00:00
static bool __usbstorage_IsInserted(void)
2012-01-21 20:57:41 +00:00
{
return (USBStorage2_GetCapacity(0, NULL) != 0);
2012-01-21 20:57:41 +00:00
}
static bool __usbstorage_ReadSectors(u32 sector, u32 numSectors, void *buffer)
2012-01-21 20:57:41 +00:00
{
return (USBStorage2_ReadSectors(0, sector, numSectors, buffer) >= 0);
2012-01-21 20:57:41 +00:00
}
static bool __usbstorage_WriteSectors(u32 sector, u32 numSectors, const void *buffer)
2012-01-21 20:57:41 +00:00
{
return (USBStorage2_WriteSectors(0, sector, numSectors, buffer) >= 0);
2012-01-21 20:57:41 +00:00
}
static bool __usbstorage_ClearStatus(void)
2012-01-21 20:57:41 +00:00
{
return true;
2012-01-21 20:57:41 +00:00
}
static bool __usbstorage_Shutdown(void)
2012-01-21 20:57:41 +00:00
{
hddInUse[0] = false;
hdd_sector_size[0] = 512;
2012-01-21 20:57:41 +00:00
return true;
}
const DISC_INTERFACE __io_usbstorage2_port0 = {
DEVICE_TYPE_WII_UMS, FEATURE_MEDIUM_CANREAD | FEATURE_MEDIUM_CANWRITE | FEATURE_WII_USB,
(FN_MEDIUM_STARTUP) &__usbstorage_Startup,
(FN_MEDIUM_ISINSERTED) &__usbstorage_IsInserted,
(FN_MEDIUM_READSECTORS) &__usbstorage_ReadSectors,
(FN_MEDIUM_WRITESECTORS) &__usbstorage_WriteSectors,
(FN_MEDIUM_CLEARSTATUS) &__usbstorage_ClearStatus,
(FN_MEDIUM_SHUTDOWN) &__usbstorage_Shutdown
};
static bool __usbstorage_Startup2(void)
{
return USBStorage2_Init(1) >= 0;
}
static bool __usbstorage_IsInserted2(void)
{
return (USBStorage2_GetCapacity(1, NULL) != 0);
}
static bool __usbstorage_ReadSectors2(u32 sector, u32 numSectors, void *buffer)
{
return (USBStorage2_ReadSectors(1, sector, numSectors, buffer) >= 0);
}
static bool __usbstorage_WriteSectors2(u32 sector, u32 numSectors, const void *buffer)
{
return (USBStorage2_WriteSectors(1, sector, numSectors, buffer) >= 0);
}
static bool __usbstorage_Shutdown2(void)
2012-01-21 20:57:41 +00:00
{
hddInUse[1] = false;
hdd_sector_size[1] = 512;
2012-01-21 20:57:41 +00:00
return true;
}
const DISC_INTERFACE __io_usbstorage2_port1 = {
DEVICE_TYPE_WII_UMS, FEATURE_MEDIUM_CANREAD | FEATURE_MEDIUM_CANWRITE | FEATURE_WII_USB,
(FN_MEDIUM_STARTUP) &__usbstorage_Startup2,
(FN_MEDIUM_ISINSERTED) &__usbstorage_IsInserted2,
(FN_MEDIUM_READSECTORS) &__usbstorage_ReadSectors2,
(FN_MEDIUM_WRITESECTORS) &__usbstorage_WriteSectors2,
(FN_MEDIUM_CLEARSTATUS) &__usbstorage_ClearStatus,
(FN_MEDIUM_SHUTDOWN) &__usbstorage_Shutdown2
};