2019-06-16 15:01:49 +00:00
|
|
|
/*
|
|
|
|
Copyright 2016-2019 Arisotura
|
|
|
|
|
|
|
|
This file is part of melonDS.
|
|
|
|
|
|
|
|
melonDS 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 3 of the License, or (at your option)
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
melonDS 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 melonDS. If not, see http://www.gnu.org/licenses/.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include "DSi.h"
|
|
|
|
#include "DSi_SD.h"
|
2019-06-17 11:24:37 +00:00
|
|
|
#include "Platform.h"
|
2019-06-16 15:01:49 +00:00
|
|
|
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
#define SD_DESC (Num?"SDIO":"SD/MMC")
|
|
|
|
|
|
|
|
|
|
|
|
DSi_SDHost::DSi_SDHost(u32 num)
|
2019-06-16 15:01:49 +00:00
|
|
|
{
|
|
|
|
Num = num;
|
2019-06-17 11:24:37 +00:00
|
|
|
|
2019-06-18 12:12:37 +00:00
|
|
|
DataFIFO[0] = new FIFO<u16>(0x100);
|
|
|
|
DataFIFO[1] = new FIFO<u16>(0x100);
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
Ports[0] = NULL;
|
|
|
|
Ports[1] = NULL;
|
2019-06-16 15:01:49 +00:00
|
|
|
}
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
DSi_SDHost::~DSi_SDHost()
|
2019-06-16 15:01:49 +00:00
|
|
|
{
|
2019-06-18 12:12:37 +00:00
|
|
|
delete DataFIFO[0];
|
|
|
|
delete DataFIFO[1];
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
if (Ports[0]) delete Ports[0];
|
|
|
|
if (Ports[1]) delete Ports[1];
|
2019-06-16 15:01:49 +00:00
|
|
|
}
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
void DSi_SDHost::Reset()
|
2019-06-16 15:01:49 +00:00
|
|
|
{
|
|
|
|
if (Num == 0)
|
|
|
|
{
|
|
|
|
PortSelect = 0x0200; // CHECKME
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
PortSelect = 0x0100; // CHECKME
|
|
|
|
}
|
2019-06-17 11:24:37 +00:00
|
|
|
|
|
|
|
SoftReset = 0x0007; // CHECKME
|
2019-06-17 16:40:45 +00:00
|
|
|
SDClock = 0;
|
2019-06-18 12:12:37 +00:00
|
|
|
SDOption = 0;
|
2019-06-17 11:24:37 +00:00
|
|
|
|
|
|
|
Command = 0;
|
|
|
|
Param = 0;
|
|
|
|
memset(ResponseBuffer, 0, sizeof(ResponseBuffer));
|
|
|
|
|
2019-06-18 12:12:37 +00:00
|
|
|
DataFIFO[0]->Clear();
|
|
|
|
DataFIFO[1]->Clear();
|
|
|
|
CurFIFO = 0;
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
IRQStatus = 0;
|
|
|
|
IRQMask = 0x8B7F031D;
|
|
|
|
|
2019-06-18 12:12:37 +00:00
|
|
|
DataCtl = 0;
|
|
|
|
Data32IRQ = 0;
|
|
|
|
DataMode = 0;
|
|
|
|
BlockCount16 = 0; BlockCount32 = 0; BlockCountInternal = 0;
|
|
|
|
BlockLen16 = 0; BlockLen32 = 0;
|
|
|
|
StopAction = 0;
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
if (Ports[0]) delete Ports[0];
|
|
|
|
if (Ports[1]) delete Ports[1];
|
|
|
|
Ports[0] = NULL;
|
|
|
|
Ports[1] = NULL;
|
|
|
|
|
|
|
|
if (Num == 0)
|
|
|
|
{
|
2019-06-17 16:40:45 +00:00
|
|
|
DSi_MMCStorage* mmc = new DSi_MMCStorage(this, true, "nand.bin");
|
|
|
|
mmc->SetCID(DSi::eMMC_CID);
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
// TODO: port 0 (SD)
|
2019-06-17 16:40:45 +00:00
|
|
|
Ports[1] = mmc;
|
2019-06-17 11:24:37 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// TODO: SDIO (wifi)
|
|
|
|
}
|
2019-06-16 15:01:49 +00:00
|
|
|
}
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
void DSi_SDHost::DoSavestate(Savestate* file)
|
2019-06-16 15:01:49 +00:00
|
|
|
{
|
|
|
|
// TODO!
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-06-18 12:12:37 +00:00
|
|
|
void DSi_SDHost::ClearIRQ(u32 irq)
|
|
|
|
{
|
|
|
|
IRQStatus &= ~(1<<irq);
|
|
|
|
}
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
void DSi_SDHost::SetIRQ(u32 irq)
|
2019-06-16 15:01:49 +00:00
|
|
|
{
|
2019-06-17 11:24:37 +00:00
|
|
|
u32 oldflags = IRQStatus & ~IRQMask;
|
|
|
|
|
|
|
|
IRQStatus |= (1<<irq);
|
|
|
|
u32 newflags = IRQStatus & ~IRQMask;
|
|
|
|
|
|
|
|
if ((oldflags == 0) && (newflags != 0))
|
|
|
|
NDS::SetIRQ2(Num ? NDS::IRQ2_DSi_SDIO : NDS::IRQ2_DSi_SDMMC);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DSi_SDHost::SendResponse(u32 val, bool last)
|
|
|
|
{
|
|
|
|
*(u32*)&ResponseBuffer[6] = *(u32*)&ResponseBuffer[4];
|
|
|
|
*(u32*)&ResponseBuffer[4] = *(u32*)&ResponseBuffer[2];
|
|
|
|
*(u32*)&ResponseBuffer[2] = *(u32*)&ResponseBuffer[0];
|
|
|
|
*(u32*)&ResponseBuffer[0] = val;
|
|
|
|
|
|
|
|
if (last) SetIRQ(0);
|
|
|
|
}
|
|
|
|
|
2019-06-18 12:12:37 +00:00
|
|
|
void DSi_SDHost::FinishSend(u32 param)
|
|
|
|
{
|
|
|
|
DSi_SDHost* host = (param & 0x1) ? DSi::SDIO : DSi::SDMMC;
|
|
|
|
|
|
|
|
host->CurFIFO ^= 1;
|
|
|
|
|
|
|
|
host->ClearIRQ(25);
|
|
|
|
host->SetIRQ(24);
|
|
|
|
if (param & 0x2) host->SetIRQ(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DSi_SDHost::SendData(u8* data, u32 len, bool last)
|
|
|
|
{
|
|
|
|
printf("%s: data RX, len=%d, blkcnt=%d blklen=%d, irq=%08X\n", SD_DESC, len, BlockCount16, BlockLen16, IRQMask);
|
|
|
|
if (len != BlockLen16) printf("!! BAD BLOCKLEN\n");
|
|
|
|
|
|
|
|
u32 f = CurFIFO ^ 1;
|
|
|
|
for (u32 i = 0; i < len; i += 2)
|
|
|
|
DataFIFO[f]->Write(*(u16*)&data[i]);
|
|
|
|
|
|
|
|
//CurFIFO = f;
|
|
|
|
//SetIRQ(24);
|
|
|
|
// TODO: determine what the delay should be!
|
|
|
|
// for now, this is a placeholder
|
|
|
|
// we need a delay because DSi boot2 will send a command and then wait for IRQ0
|
|
|
|
// but if IRQ24 is thrown instantly, the handler clears IRQ0 before the
|
|
|
|
// send-command function starts polling IRQ status
|
|
|
|
u32 param = Num | (last << 1);
|
|
|
|
NDS::ScheduleEvent(NDS::Event_DSi_SDTransfer, false, 512, FinishSend, param);
|
|
|
|
}
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
|
|
|
|
u16 DSi_SDHost::Read(u32 addr)
|
|
|
|
{
|
|
|
|
//printf("SDMMC READ %08X %08X\n", addr, NDS::GetPC(1));
|
|
|
|
|
2019-06-16 15:01:49 +00:00
|
|
|
switch (addr & 0x1FF)
|
|
|
|
{
|
2019-06-17 11:24:37 +00:00
|
|
|
case 0x000: return Command;
|
2019-06-16 15:01:49 +00:00
|
|
|
case 0x002: return PortSelect & 0x030F;
|
2019-06-17 11:24:37 +00:00
|
|
|
case 0x004: return Param & 0xFFFF;
|
|
|
|
case 0x006: return Param >> 16;
|
|
|
|
|
2019-06-18 12:12:37 +00:00
|
|
|
case 0x008: return StopAction;
|
|
|
|
case 0x00A: return BlockCount16;
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
case 0x00C: return ResponseBuffer[0];
|
|
|
|
case 0x00E: return ResponseBuffer[1];
|
|
|
|
case 0x010: return ResponseBuffer[2];
|
|
|
|
case 0x012: return ResponseBuffer[3];
|
|
|
|
case 0x014: return ResponseBuffer[4];
|
|
|
|
case 0x016: return ResponseBuffer[5];
|
|
|
|
case 0x018: return ResponseBuffer[6];
|
|
|
|
case 0x01A: return ResponseBuffer[7];
|
|
|
|
|
2019-06-18 12:12:37 +00:00
|
|
|
case 0x01C: return (IRQStatus & 0x031D) | 0x0030; // TODO: adjust insert flags for SD card
|
|
|
|
case 0x01E: return ((IRQStatus >> 16) & 0x8B7F);
|
2019-06-17 11:24:37 +00:00
|
|
|
case 0x020: return IRQMask & 0x031D;
|
|
|
|
case 0x022: return (IRQMask >> 16) & 0x8B7F;
|
|
|
|
|
2019-06-17 16:40:45 +00:00
|
|
|
case 0x024: return SDClock;
|
2019-06-18 12:12:37 +00:00
|
|
|
case 0x026: return BlockLen16;
|
|
|
|
case 0x028: return SDOption;
|
|
|
|
|
|
|
|
case 0x030: // FIFO16
|
|
|
|
{
|
|
|
|
// TODO: decrement BlockLen????
|
|
|
|
|
|
|
|
u32 f = CurFIFO;
|
|
|
|
if (DataFIFO[f]->IsEmpty())
|
|
|
|
{
|
|
|
|
// TODO
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
DSi_SDDevice* dev = Ports[PortSelect & 0x1];
|
|
|
|
u16 ret = DataFIFO[f]->Read();
|
|
|
|
|
|
|
|
if (DataFIFO[f]->IsEmpty())
|
|
|
|
{
|
|
|
|
ClearIRQ(24);
|
|
|
|
|
|
|
|
if (BlockCountInternal == 0)
|
|
|
|
{
|
|
|
|
printf("%s: data RX complete", SD_DESC);
|
|
|
|
|
|
|
|
if (StopAction & (1<<8))
|
|
|
|
{
|
|
|
|
printf(", sending CMD12");
|
|
|
|
if (dev) dev->SendCMD(12, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("\n");
|
|
|
|
|
|
|
|
// CHECKME: presumably IRQ2 should not trigger here, but rather
|
|
|
|
// when the data transfer is done
|
|
|
|
//SetIRQ(0);
|
|
|
|
//SetIRQ(2);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
BlockCountInternal--;
|
|
|
|
|
|
|
|
if (dev) dev->ContinueTransfer();
|
|
|
|
}
|
|
|
|
|
|
|
|
SetIRQ(25);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 0x0D8: return DataCtl;
|
2019-06-17 16:40:45 +00:00
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
case 0x0E0: return SoftReset;
|
2019-06-18 12:12:37 +00:00
|
|
|
|
|
|
|
case 0x100: return Data32IRQ;
|
|
|
|
case 0x104: return BlockLen32;
|
|
|
|
case 0x108: return BlockCount32;
|
2019-06-16 15:01:49 +00:00
|
|
|
}
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
printf("unknown %s read %08X\n", SD_DESC, addr);
|
2019-06-16 15:01:49 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
void DSi_SDHost::Write(u32 addr, u16 val)
|
2019-06-16 15:01:49 +00:00
|
|
|
{
|
2019-06-17 11:24:37 +00:00
|
|
|
//printf("SDMMC WRITE %08X %04X %08X\n", addr, val, NDS::GetPC(1));
|
|
|
|
|
2019-06-16 15:01:49 +00:00
|
|
|
switch (addr & 0x1FF)
|
|
|
|
{
|
2019-06-17 11:24:37 +00:00
|
|
|
case 0x000:
|
|
|
|
{
|
|
|
|
Command = val;
|
|
|
|
u8 cmd = Command & 0x3F;
|
|
|
|
|
|
|
|
DSi_SDDevice* dev = Ports[PortSelect & 0x1];
|
|
|
|
if (dev)
|
|
|
|
{
|
2019-06-17 16:40:45 +00:00
|
|
|
// CHECKME
|
|
|
|
// "Setting Command Type to "ACMD" is automatically sending an APP_CMD prefix prior to the command number"
|
|
|
|
// except DSi boot2 manually sends an APP_CMD prefix AND sets the next command to be ACMD
|
2019-06-17 11:24:37 +00:00
|
|
|
switch ((Command >> 6) & 0x3)
|
|
|
|
{
|
|
|
|
case 0: dev->SendCMD(cmd, Param); break;
|
2019-06-17 16:40:45 +00:00
|
|
|
case 1: /*dev->SendCMD(55, 0);*/ dev->SendCMD(cmd, Param); break;
|
2019-06-17 11:24:37 +00:00
|
|
|
default:
|
|
|
|
printf("%s: unknown command type %d, %02X %08X\n", SD_DESC, (Command>>6)&0x3, cmd, Param);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
|
2019-06-18 12:12:37 +00:00
|
|
|
case 0x002: PortSelect = val; printf("%s: PORT SELECT %04X\n", SD_DESC, val); return;
|
2019-06-17 11:24:37 +00:00
|
|
|
case 0x004: Param = (Param & 0xFFFF0000) | val; return;
|
|
|
|
case 0x006: Param = (Param & 0x0000FFFF) | (val << 16); return;
|
|
|
|
|
2019-06-18 12:12:37 +00:00
|
|
|
case 0x008: StopAction = val & 0x0101; return;
|
|
|
|
case 0x00A: BlockCount16 = val; BlockCountInternal = val; return;
|
|
|
|
|
|
|
|
case 0x01C: IRQStatus &= (val | 0xFFFF0000); return;
|
|
|
|
case 0x01E: IRQStatus &= ((val << 16) | 0xFFFF); return;
|
2019-06-17 11:24:37 +00:00
|
|
|
case 0x020: IRQMask = (IRQMask & 0x8B7F0000) | (val & 0x031D); return;
|
|
|
|
case 0x022: IRQMask = (IRQMask & 0x0000031D) | ((val & 0x8B7F) << 16); return;
|
|
|
|
|
2019-06-17 16:40:45 +00:00
|
|
|
case 0x024: SDClock = val & 0x03FF; return;
|
2019-06-18 12:12:37 +00:00
|
|
|
case 0x026:
|
|
|
|
BlockLen16 = val & 0x03FF;
|
|
|
|
if (BlockLen16 > 0x200) BlockLen16 = 0x200;
|
|
|
|
return;
|
|
|
|
case 0x028: SDOption = val & 0xC1FF; return;
|
|
|
|
|
|
|
|
case 0x0D8:
|
|
|
|
DataCtl = (val & 0x0022);
|
|
|
|
DataMode = ((DataCtl >> 1) & 0x1) & ((Data32IRQ >> 1) & 0x1);
|
|
|
|
printf("%s: data mode %d-bit\n", SD_DESC, DataMode?32:16);
|
|
|
|
return;
|
2019-06-17 16:40:45 +00:00
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
case 0x0E0:
|
|
|
|
if ((SoftReset & 0x0001) && !(val & 0x0001))
|
|
|
|
{
|
|
|
|
printf("%s: RESET\n", SD_DESC);
|
2019-06-18 12:12:37 +00:00
|
|
|
StopAction = 0;
|
|
|
|
memset(ResponseBuffer, 0, sizeof(ResponseBuffer));
|
2019-06-17 11:24:37 +00:00
|
|
|
IRQStatus = 0;
|
|
|
|
// TODO: ERROR_DETAIL_STATUS
|
2019-06-18 12:12:37 +00:00
|
|
|
SDClock &= ~0x0500;
|
|
|
|
SDOption = 0x40EE;
|
2019-06-17 11:24:37 +00:00
|
|
|
// TODO: CARD_IRQ_STAT
|
|
|
|
// TODO: FIFO16 shit
|
|
|
|
}
|
|
|
|
SoftReset = 0x0006 | (val & 0x0001);
|
|
|
|
return;
|
2019-06-18 12:12:37 +00:00
|
|
|
|
|
|
|
case 0x100:
|
|
|
|
Data32IRQ = (val & 0x1802) | (Data32IRQ & 0x0300);
|
|
|
|
if (val & (1<<10)) printf("TODO: SD/MMC: CLEAR FIFO32\n");
|
|
|
|
DataMode = ((DataCtl >> 1) & 0x1) & ((Data32IRQ >> 1) & 0x1);
|
|
|
|
printf("%s: data mode %d-bit\n", SD_DESC, DataMode?32:16);
|
|
|
|
return;
|
|
|
|
case 0x104: BlockLen32 = val & 0x03FF; return;
|
|
|
|
case 0x108: BlockCount32 = val; return;
|
2019-06-17 11:24:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
printf("unknown %s write %08X %04X\n", SD_DESC, addr, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DSi_MMCStorage::DSi_MMCStorage(DSi_SDHost* host, bool internal, const char* path) : DSi_SDDevice(host)
|
|
|
|
{
|
|
|
|
Internal = internal;
|
|
|
|
strncpy(FilePath, path, 1023); FilePath[1023] = '\0';
|
|
|
|
|
|
|
|
File = Platform::OpenLocalFile(path, "r+b");
|
|
|
|
|
2019-06-18 12:12:37 +00:00
|
|
|
CSR = 0x00000100; // checkme
|
2019-06-17 16:40:45 +00:00
|
|
|
|
|
|
|
// TODO: busy bit
|
|
|
|
// TODO: SDHC/SDXC bit
|
|
|
|
OCR = 0x80FF8000;
|
|
|
|
|
|
|
|
// TODO: customize based on card size etc
|
|
|
|
u8 csd_template[16] = {0x40, 0x40, 0x96, 0xE9, 0x7F, 0xDB, 0xF6, 0xDF, 0x01, 0x59, 0x0F, 0x2A, 0x01, 0x26, 0x90, 0x00};
|
|
|
|
memcpy(CSD, csd_template, 16);
|
2019-06-18 12:12:37 +00:00
|
|
|
|
|
|
|
// checkme
|
|
|
|
memset(SCR, 0, 8);
|
|
|
|
*(u32*)&SCR[0] = 0x012A0000;
|
|
|
|
|
|
|
|
memset(SSR, 0, 64);
|
2019-06-17 11:24:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DSi_MMCStorage::~DSi_MMCStorage()
|
|
|
|
{
|
|
|
|
if (File) fclose(File);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DSi_MMCStorage::SendCMD(u8 cmd, u32 param)
|
|
|
|
{
|
2019-06-17 16:40:45 +00:00
|
|
|
if (CSR & (1<<5))
|
|
|
|
{
|
|
|
|
CSR &= ~(1<<5);
|
|
|
|
return SendACMD(cmd, param);
|
|
|
|
}
|
|
|
|
|
2019-06-17 11:24:37 +00:00
|
|
|
switch (cmd)
|
|
|
|
{
|
2019-06-17 16:40:45 +00:00
|
|
|
case 0: // reset/etc
|
2019-06-17 11:24:37 +00:00
|
|
|
Host->SendResponse(CSR, true);
|
|
|
|
return;
|
|
|
|
|
2019-06-17 16:40:45 +00:00
|
|
|
case 2:
|
|
|
|
case 10: // get CID
|
|
|
|
Host->SendResponse(*(u32*)&CID[0], false);
|
|
|
|
Host->SendResponse(*(u32*)&CID[1], false);
|
|
|
|
Host->SendResponse(*(u32*)&CID[2], false);
|
|
|
|
Host->SendResponse(*(u32*)&CID[3], true);
|
|
|
|
//if (cmd == 2) SetState(0x02);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 3: // get/set RCA
|
|
|
|
if (Internal)
|
|
|
|
{
|
|
|
|
RCA = param >> 16;
|
|
|
|
Host->SendResponse(CSR|0x10000, true); // huh??
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// TODO
|
|
|
|
printf("CMD3 on SD card: TODO\n");
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 7: // select card (by RCA)
|
|
|
|
Host->SendResponse(CSR, true);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 8: // set voltage
|
2019-06-17 11:24:37 +00:00
|
|
|
Host->SendResponse(param, true);
|
|
|
|
return;
|
2019-06-17 16:40:45 +00:00
|
|
|
|
|
|
|
case 9: // get CSD
|
|
|
|
Host->SendResponse(*(u32*)&CSD[0], false);
|
|
|
|
Host->SendResponse(*(u32*)&CSD[1], false);
|
|
|
|
Host->SendResponse(*(u32*)&CSD[2], false);
|
|
|
|
Host->SendResponse(*(u32*)&CSD[3], true);
|
|
|
|
return;
|
|
|
|
|
2019-06-18 12:12:37 +00:00
|
|
|
case 12: // stop operation
|
|
|
|
Host->SendResponse(CSR, true);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 16: // set block size
|
|
|
|
BlockSize = param;
|
|
|
|
Host->SendResponse(CSR, true);
|
|
|
|
return;
|
|
|
|
|
2019-06-17 16:40:45 +00:00
|
|
|
case 55: // ??
|
|
|
|
CSR |= (1<<5);
|
|
|
|
Host->SendResponse(CSR, true);
|
|
|
|
return;
|
2019-06-16 15:01:49 +00:00
|
|
|
}
|
|
|
|
|
2019-06-17 16:40:45 +00:00
|
|
|
printf("MMC: unknown CMD %d %08X\n", cmd, param);
|
2019-06-17 11:24:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DSi_MMCStorage::SendACMD(u8 cmd, u32 param)
|
|
|
|
{
|
2019-06-17 16:40:45 +00:00
|
|
|
switch (cmd)
|
|
|
|
{
|
2019-06-18 12:12:37 +00:00
|
|
|
case 6: // set bus width (TODO?)
|
|
|
|
printf("SET BUS WIDTH %08X\n", param);
|
|
|
|
Host->SendResponse(CSR, true);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 13: // get SSR
|
|
|
|
Host->SendResponse(CSR, true);
|
|
|
|
Host->SendData(SSR, 64, true);
|
|
|
|
return;
|
|
|
|
|
2019-06-17 16:40:45 +00:00
|
|
|
case 41: // set operating conditions
|
|
|
|
OCR &= 0xBF000000;
|
|
|
|
OCR |= (param & 0x40FFFFFF);
|
|
|
|
Host->SendResponse(OCR, true);
|
|
|
|
SetState(0x01);
|
|
|
|
return;
|
2019-06-18 12:12:37 +00:00
|
|
|
|
|
|
|
case 42: // ???
|
|
|
|
Host->SendResponse(CSR, true);
|
|
|
|
return;
|
|
|
|
|
|
|
|
case 51: // get SCR
|
|
|
|
Host->SendResponse(CSR, true);
|
|
|
|
Host->SendData(SCR, 8, true);
|
|
|
|
return;
|
2019-06-17 16:40:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
printf("MMC: unknown ACMD %d %08X\n", cmd, param);
|
2019-06-16 15:01:49 +00:00
|
|
|
}
|
2019-06-18 12:12:37 +00:00
|
|
|
|
|
|
|
void DSi_MMCStorage::ContinueTransfer()
|
|
|
|
{
|
|
|
|
//
|
|
|
|
}
|