2015-01-17 10:11:08 +00:00

195 lines
4.3 KiB
C

/*
* Copyright (C) 2010 Spaceman Spiff
* Copyright (C) 2010 Hermes
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
#include <types.h>
#include <module.h>
#include <syscalls.h>
#define DVDIO_SIZE 32
#define DVDIO_ADDR 0xD006000
#define SECTORSIZE 2048
#define MAX_SIZE_DI (0xFF << 15)
static u32 DIP_adjust_read_size(u8 *buf, u32 size, u32 align);
int DIP_StopMotor(void)
{
u32 inbuf[8];
inbuf[0] = IOCTL_DI_STOPMOTOR << 24;
inbuf[1] = 0;
inbuf[2] = 0;
return handleDiCommand(inbuf,NULL,0);
}
int DIP_CustomCommand(u8 *cmd, u8 *ans)
{
volatile unsigned long *dvdio = (volatile unsigned long *) DVDIO_ADDR;
dip_memcpy((u8 *) dvdio, cmd, DVDIO_SIZE);
os_sync_after_write((u8 *) dvdio, DVDIO_SIZE);
// DI Control Register, wait until ready
while (dvdio[7] & 1) // 0xD00601C
;
dip_memcpy(ans, (u8 *) dvdio, DVDIO_SIZE);
os_sync_after_write(ans, DVDIO_SIZE);
return dvdio[8]; // 0xD006020
}
int DIP_ReadDVDVideo(void *dst, u32 len, u32 lba)
{
int res;
int tries = 0;
u32 inbuf[8];
do {
inbuf[0] = IOCTL_DI_READDVD << 24;
inbuf[1] = 0;
inbuf[2] = 0;
inbuf[3] = len >> 11; // Convert to 2048 blocks (dvd sector)
inbuf[4] = lba;
os_sync_before_read(dst, len);
res = handleDiCommand(inbuf, dst, len);
tries++;
} while (res && tries < 32);
return res;
}
int DIP_ReadDVD(void *dst, u32 len, u32 sector)
{
u32 inbuf[8];
int res;
int tries = 0;
do {
inbuf[0] = IOCTL_DI_LOWREAD << 24; // IOCTL_DI_LOWREAD A8
inbuf[1] = len;
inbuf[2] = sector;
os_sync_before_read(dst, len);
res =handleDiCommand(inbuf, dst, len);
tries++;
} while (res && tries < 8);
return res;
}
int DIP_ReadDVDRom(u8 *outbuf, u32 len, u32 offset)
{
u32 cnt = 0;
int res;
if (len == 0)
return 0;
u32 lba = offset >> 9; // offset / 512
u32 size;
while (cnt < len) {
u32 skip;
size = len - cnt;
skip = (offset > (lba << 9)) ? (offset - (lba << 9)) << 2 : 0;
res = DIP_adjust_read_size(outbuf+ cnt, size, SECTORSIZE);
if (skip || !res) {
if ((skip + size) > SECTORSIZE)
size = SECTORSIZE - skip;
u8 *mem = dip_alloc_aligned(SECTORSIZE, 0x20);
if (!mem)
return -1;
res = DIP_ReadDVDVideo(mem, SECTORSIZE, lba);
if (res == 0) {
dip_memcpy(outbuf+cnt, mem + skip, size);
os_sync_after_write(outbuf+cnt, size);
}
dip_free(mem);
} else {
size = res;
if (size >= MAX_SIZE_DI)
size = MAX_SIZE_DI;
res = DIP_ReadDVDVideo(outbuf+cnt, size, lba); // inlined function
/*
u32 tries = 0;
do {
u32 inbuf[8];
inbuf[0] = IOCTL_DI_READDVD << 24;
inbuf[1] = 0;
inbuf[2] = 0;
inbuf[3] = size >> 11;
inbuf[4] = lba;
os_sync_before_read(outbuf, size);
res = handleDiCommand(inbuf, (u32 *) outbuf, size);
tries = tries + 1;
} while (res != 0 && tries < 15);
*/
}
if (res)
return res;
cnt += size;
lba += (size + skip) >> 11;
}
return res;
}
/* si retorna cero, el buffer, no esta alineado. */
// desde dip_plugin
#define DMA1_START_ADDRESS 0x00000000
#define DMA1_END_ADDRESS 0x01800000
#define DMA2_START_ADDRESS 0x10000000
#define DMA2_END_ADDRESS 0x13618000
u32 DIP_adjust_read_size(u8 *outbuf, u32 size, u32 alignment)
{
u32 mem;
int ret = 0;
/* Output buffer address */
mem = (u32)outbuf;
/* Check for memory alignment */
if (!(mem & (0x1f))) {
u32 dmalen = 0;
/* DMA1 range check */
if ((mem >= DMA1_START_ADDRESS) && (mem < DMA1_END_ADDRESS))
dmalen = (DMA1_END_ADDRESS - mem);
/* DMA2 range check */
if ((mem >= DMA2_START_ADDRESS) && (mem < DMA2_END_ADDRESS))
dmalen = (DMA2_END_ADDRESS - mem);
if (dmalen >= alignment)
ret = (dmalen < size) ? dmalen : size;
ret -= (ret & (alignment-1));
}
return ret;
}