2016-12-05 17:02:29 +00:00
|
|
|
/*
|
|
|
|
Copyright 2016-2017 StapleButter
|
|
|
|
|
|
|
|
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/.
|
|
|
|
*/
|
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
#include <stdio.h>
|
2017-01-20 00:18:30 +00:00
|
|
|
#include <string.h>
|
2016-12-04 02:20:50 +00:00
|
|
|
#include "NDS.h"
|
|
|
|
#include "SPI.h"
|
|
|
|
|
|
|
|
|
|
|
|
namespace SPI_Firmware
|
|
|
|
{
|
|
|
|
|
|
|
|
u8* Firmware;
|
|
|
|
u32 FirmwareLength;
|
|
|
|
|
|
|
|
u32 Hold;
|
|
|
|
u8 CurCmd;
|
|
|
|
u32 DataPos;
|
|
|
|
u8 Data;
|
|
|
|
|
2016-12-05 17:02:29 +00:00
|
|
|
u8 StatusReg;
|
2016-12-04 02:20:50 +00:00
|
|
|
u32 Addr;
|
|
|
|
|
2017-01-16 03:47:37 +00:00
|
|
|
|
|
|
|
u16 CRC16(u8* data, u32 len, u32 start)
|
|
|
|
{
|
|
|
|
u16 blarg[8] = {0xC0C1, 0xC181, 0xC301, 0xC601, 0xCC01, 0xD801, 0xF001, 0xA001};
|
|
|
|
|
|
|
|
for (u32 i = 0; i < len; i++)
|
|
|
|
{
|
|
|
|
start ^= data[i];
|
|
|
|
|
|
|
|
for (int j = 0; j < 8; j++)
|
|
|
|
{
|
|
|
|
if (start & 0x1)
|
|
|
|
{
|
|
|
|
start >>= 1;
|
|
|
|
start ^= (blarg[j] << (7-j));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
start >>= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return start & 0xFFFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VerifyCRC16(u32 start, u32 offset, u32 len, u32 crcoffset)
|
|
|
|
{
|
|
|
|
u16 crc_stored = *(u16*)&Firmware[crcoffset];
|
|
|
|
u16 crc_calced = CRC16(&Firmware[offset], len, start);
|
|
|
|
//printf("%04X vs %04X\n", crc_stored, crc_calced);
|
|
|
|
return (crc_stored == crc_calced);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
void Init()
|
|
|
|
{
|
|
|
|
Firmware = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Reset()
|
|
|
|
{
|
|
|
|
if (Firmware) delete[] Firmware;
|
|
|
|
|
|
|
|
FILE* f = fopen("firmware.bin", "rb");
|
|
|
|
fseek(f, 0, SEEK_END);
|
|
|
|
FirmwareLength = (u32)ftell(f);
|
|
|
|
Firmware = new u8[FirmwareLength];
|
|
|
|
|
|
|
|
fseek(f, 0, SEEK_SET);
|
|
|
|
fread(Firmware, FirmwareLength, 1, f);
|
|
|
|
|
|
|
|
fclose(f);
|
|
|
|
|
2017-01-30 17:36:11 +00:00
|
|
|
// temp: disable autoboot
|
|
|
|
/*Firmware[0x3FE64] &= 0xBF;
|
|
|
|
*(u16*)&Firmware[0x3FE72] = CRC16(&Firmware[0x3FE00], 0x70, 0xFFFF);
|
|
|
|
Firmware[0x3FF64] &= 0xBF;
|
|
|
|
*(u16*)&Firmware[0x3FF72] = CRC16(&Firmware[0x3FF00], 0x70, 0xFFFF);*/
|
|
|
|
|
2017-01-16 03:47:37 +00:00
|
|
|
// verify shit
|
|
|
|
printf("FW: WIFI CRC16 = %s\n", VerifyCRC16(0x0000, 0x2C, *(u16*)&Firmware[0x2C], 0x2A)?"GOOD":"BAD");
|
|
|
|
printf("FW: AP1 CRC16 = %s\n", VerifyCRC16(0x0000, 0x3FA00, 0xFE, 0x3FAFE)?"GOOD":"BAD");
|
|
|
|
printf("FW: AP2 CRC16 = %s\n", VerifyCRC16(0x0000, 0x3FB00, 0xFE, 0x3FBFE)?"GOOD":"BAD");
|
|
|
|
printf("FW: AP3 CRC16 = %s\n", VerifyCRC16(0x0000, 0x3FC00, 0xFE, 0x3FCFE)?"GOOD":"BAD");
|
|
|
|
printf("FW: USER0 CRC16 = %s\n", VerifyCRC16(0xFFFF, 0x3FE00, 0x70, 0x3FE72)?"GOOD":"BAD");
|
|
|
|
printf("FW: USER1 CRC16 = %s\n", VerifyCRC16(0xFFFF, 0x3FF00, 0x70, 0x3FF72)?"GOOD":"BAD");
|
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
Hold = 0;
|
|
|
|
CurCmd = 0;
|
|
|
|
Data = 0;
|
2016-12-05 17:02:29 +00:00
|
|
|
StatusReg = 0x00;
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
u8 Read()
|
|
|
|
{
|
|
|
|
return Data;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Write(u8 val, u32 hold)
|
|
|
|
{
|
|
|
|
if (!hold)
|
|
|
|
{
|
|
|
|
Hold = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hold && (!Hold))
|
|
|
|
{
|
|
|
|
CurCmd = val;
|
|
|
|
Hold = 1;
|
2016-12-05 17:02:29 +00:00
|
|
|
Data = 0;
|
2016-12-04 02:20:50 +00:00
|
|
|
DataPos = 1;
|
|
|
|
Addr = 0;
|
2016-12-05 22:17:03 +00:00
|
|
|
//printf("firmware SPI command %02X\n", CurCmd);
|
2016-12-04 02:20:50 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (CurCmd)
|
|
|
|
{
|
|
|
|
case 0x03: // read
|
|
|
|
{
|
|
|
|
if (DataPos < 4)
|
|
|
|
{
|
|
|
|
Addr <<= 8;
|
|
|
|
Addr |= val;
|
|
|
|
Data = 0;
|
|
|
|
|
2016-12-05 22:17:03 +00:00
|
|
|
//if (DataPos == 3) printf("firmware SPI read %08X\n", Addr);
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (Addr >= FirmwareLength)
|
|
|
|
Data = 0;
|
|
|
|
else
|
|
|
|
Data = Firmware[Addr];
|
|
|
|
|
|
|
|
Addr++;
|
|
|
|
}
|
|
|
|
|
|
|
|
DataPos++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2016-12-05 17:02:29 +00:00
|
|
|
case 0x04: // write disable
|
|
|
|
StatusReg &= ~(1<<1);
|
|
|
|
Data = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x05: // read status reg
|
|
|
|
Data = StatusReg;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x06: // write enable
|
|
|
|
StatusReg |= (1<<1);
|
|
|
|
Data = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x9F: // read JEDEC ID
|
|
|
|
{
|
|
|
|
switch (DataPos)
|
|
|
|
{
|
|
|
|
case 1: Data = 0x20; break;
|
|
|
|
case 2: Data = 0x40; break;
|
|
|
|
case 3: Data = 0x12; break;
|
|
|
|
default: Data = 0; break;
|
|
|
|
}
|
|
|
|
DataPos++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
default:
|
|
|
|
printf("unknown firmware SPI command %02X\n", CurCmd);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
namespace SPI_Powerman
|
|
|
|
{
|
|
|
|
|
|
|
|
u32 Hold;
|
|
|
|
u32 DataPos;
|
|
|
|
u8 Index;
|
|
|
|
u8 Data;
|
|
|
|
|
|
|
|
u8 Registers[8];
|
|
|
|
u8 RegMasks[8];
|
|
|
|
|
|
|
|
|
|
|
|
void Init()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void Reset()
|
|
|
|
{
|
|
|
|
Hold = 0;
|
|
|
|
Index = 0;
|
|
|
|
Data = 0;
|
|
|
|
|
|
|
|
memset(Registers, 0, sizeof(Registers));
|
|
|
|
memset(RegMasks, 0, sizeof(RegMasks));
|
|
|
|
|
|
|
|
Registers[4] = 0x40;
|
|
|
|
|
|
|
|
RegMasks[0] = 0x7F;
|
|
|
|
RegMasks[1] = 0x01;
|
|
|
|
RegMasks[2] = 0x01;
|
|
|
|
RegMasks[3] = 0x03;
|
|
|
|
RegMasks[4] = 0x0F;
|
|
|
|
}
|
|
|
|
|
|
|
|
u8 Read()
|
|
|
|
{
|
|
|
|
return Data;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Write(u8 val, u32 hold)
|
|
|
|
{
|
|
|
|
if (!hold)
|
|
|
|
{
|
|
|
|
Hold = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hold && (!Hold))
|
|
|
|
{
|
|
|
|
Index = val;
|
|
|
|
Hold = 1;
|
|
|
|
Data = 0;
|
|
|
|
DataPos = 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (DataPos == 1)
|
|
|
|
{
|
|
|
|
if (Index & 0x80)
|
|
|
|
{
|
|
|
|
Data = Registers[Index & 0x07];
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Registers[Index & 0x07] =
|
|
|
|
(Registers[Index & 0x07] & ~RegMasks[Index & 0x07]) |
|
|
|
|
(val & RegMasks[Index & 0x07]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
Data = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
namespace SPI
|
|
|
|
{
|
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
u16 Cnt;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
u32 CurDevice;
|
|
|
|
|
|
|
|
|
|
|
|
void Init()
|
|
|
|
{
|
|
|
|
SPI_Firmware::Init();
|
2017-01-20 00:18:30 +00:00
|
|
|
SPI_Powerman::Init();
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Reset()
|
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
Cnt = 0;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
SPI_Firmware::Reset();
|
2017-01-20 00:18:30 +00:00
|
|
|
SPI_Powerman::Reset();
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void WriteCnt(u16 val)
|
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
Cnt = (Cnt & 0x0080) | (val & 0xCF03);
|
2016-12-04 02:20:50 +00:00
|
|
|
if (val & 0x0400) printf("!! CRAPOED 16BIT SPI MODE\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
u8 ReadData()
|
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
if (!(Cnt & (1<<15))) return 0;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
switch (Cnt & 0x0300)
|
2016-12-04 02:20:50 +00:00
|
|
|
{
|
2017-01-20 00:18:30 +00:00
|
|
|
case 0x0000: return SPI_Powerman::Read();
|
2016-12-04 02:20:50 +00:00
|
|
|
case 0x0100: return SPI_Firmware::Read();
|
|
|
|
default: return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void WriteData(u8 val)
|
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
if (!(Cnt & (1<<15))) return;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
// TODO: take delays into account
|
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
switch (Cnt & 0x0300)
|
2016-12-04 02:20:50 +00:00
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
case 0x0000: SPI_Powerman::Write(val, Cnt&(1<<11)); break;
|
|
|
|
case 0x0100: SPI_Firmware::Write(val, Cnt&(1<<11)); break;
|
|
|
|
default: printf("SPI to unknown device %04X %02X\n", Cnt, val); break;
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
|
2017-01-31 16:34:17 +00:00
|
|
|
if (Cnt & (1<<14))
|
2016-12-04 02:20:50 +00:00
|
|
|
NDS::TriggerIRQ(1, NDS::IRQ_SPI);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|