2016-12-05 17:02:29 +00:00
|
|
|
/*
|
2021-03-12 19:07:24 +00:00
|
|
|
Copyright 2016-2021 Arisotura
|
2016-12-05 17:02:29 +00:00
|
|
|
|
|
|
|
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>
|
2017-05-11 21:43:57 +00:00
|
|
|
#include <stdlib.h>
|
2021-10-02 18:16:27 +00:00
|
|
|
#include <string>
|
|
|
|
#include <algorithm>
|
|
|
|
#include <codecvt>
|
|
|
|
#include <locale>
|
2016-12-04 02:20:50 +00:00
|
|
|
#include "NDS.h"
|
2021-08-30 18:26:49 +00:00
|
|
|
#include "DSi.h"
|
2016-12-04 02:20:50 +00:00
|
|
|
#include "SPI.h"
|
2019-08-04 09:44:36 +00:00
|
|
|
#include "DSi_SPI_TSC.h"
|
2019-03-27 12:34:26 +00:00
|
|
|
#include "Platform.h"
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
namespace SPI_Firmware
|
|
|
|
{
|
|
|
|
|
2021-11-17 19:42:11 +00:00
|
|
|
std::string FirmwarePath;
|
2016-12-04 02:20:50 +00:00
|
|
|
u8* Firmware;
|
|
|
|
u32 FirmwareLength;
|
2017-03-20 23:53:04 +00:00
|
|
|
u32 FirmwareMask;
|
|
|
|
|
|
|
|
u32 UserSettings;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
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);
|
|
|
|
return (crc_stored == crc_calced);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-02-07 21:23:46 +00:00
|
|
|
bool Init()
|
2016-12-04 02:20:50 +00:00
|
|
|
{
|
2021-11-17 19:42:11 +00:00
|
|
|
FirmwarePath = "";
|
|
|
|
Firmware = nullptr;
|
2017-02-07 21:23:46 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeInit()
|
|
|
|
{
|
|
|
|
if (Firmware) delete[] Firmware;
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
|
2021-01-22 18:05:07 +00:00
|
|
|
u32 FixFirmwareLength(u32 originalLength)
|
|
|
|
{
|
|
|
|
if (originalLength != 0x20000 && originalLength != 0x40000 && originalLength != 0x80000)
|
|
|
|
{
|
|
|
|
printf("Bad firmware size %d, ", originalLength);
|
|
|
|
|
|
|
|
// pick the nearest power-of-two length
|
|
|
|
originalLength |= (originalLength >> 1);
|
|
|
|
originalLength |= (originalLength >> 2);
|
|
|
|
originalLength |= (originalLength >> 4);
|
|
|
|
originalLength |= (originalLength >> 8);
|
|
|
|
originalLength |= (originalLength >> 16);
|
|
|
|
originalLength++;
|
|
|
|
|
|
|
|
// ensure it's a sane length
|
|
|
|
if (originalLength > 0x80000) originalLength = 0x80000;
|
|
|
|
else if (originalLength < 0x20000) originalLength = 0x20000;
|
|
|
|
|
|
|
|
printf("assuming %d\n", originalLength);
|
|
|
|
}
|
|
|
|
return originalLength;
|
|
|
|
}
|
|
|
|
|
2021-10-02 18:16:27 +00:00
|
|
|
void LoadDefaultFirmware()
|
2016-12-04 02:20:50 +00:00
|
|
|
{
|
2021-10-02 18:16:27 +00:00
|
|
|
FirmwareLength = 0x20000;
|
|
|
|
Firmware = new u8[FirmwareLength];
|
|
|
|
memset(Firmware, 0xFF, FirmwareLength);
|
|
|
|
FirmwareMask = FirmwareLength - 1;
|
2016-12-04 02:20:50 +00:00
|
|
|
|
2021-10-02 18:16:27 +00:00
|
|
|
u32 userdata = 0x7FE00 & FirmwareMask;
|
2020-05-17 02:37:44 +00:00
|
|
|
|
2021-10-02 18:16:27 +00:00
|
|
|
memset(Firmware + userdata, 0, 0x74);
|
2017-02-07 21:23:46 +00:00
|
|
|
|
2021-10-02 18:16:27 +00:00
|
|
|
// user settings offset
|
|
|
|
*(u16*)&Firmware[0x20] = (FirmwareLength - 0x200) >> 3;
|
|
|
|
|
|
|
|
Firmware[userdata+0x00] = 5; // version
|
|
|
|
}
|
2017-02-07 21:23:46 +00:00
|
|
|
|
2021-10-02 18:16:27 +00:00
|
|
|
void LoadFirmwareFromFile(FILE* f)
|
|
|
|
{
|
2016-12-04 02:20:50 +00:00
|
|
|
fseek(f, 0, SEEK_END);
|
2017-03-20 23:53:04 +00:00
|
|
|
|
2021-01-22 18:05:07 +00:00
|
|
|
FirmwareLength = FixFirmwareLength((u32)ftell(f));
|
2017-03-20 23:53:04 +00:00
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
Firmware = new u8[FirmwareLength];
|
|
|
|
|
|
|
|
fseek(f, 0, SEEK_SET);
|
2017-03-20 23:53:04 +00:00
|
|
|
fread(Firmware, 1, FirmwareLength, f);
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
fclose(f);
|
|
|
|
|
2017-06-25 22:35:19 +00:00
|
|
|
// take a backup
|
2021-11-17 19:42:11 +00:00
|
|
|
std::string fwBackupPath = FirmwarePath + ".bak";
|
2021-10-02 18:16:27 +00:00
|
|
|
f = Platform::OpenLocalFile(fwBackupPath, "rb");
|
|
|
|
if (!f)
|
|
|
|
{
|
|
|
|
f = Platform::OpenLocalFile(fwBackupPath, "wb");
|
|
|
|
if (f)
|
|
|
|
{
|
|
|
|
fwrite(Firmware, 1, FirmwareLength, f);
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("Could not write firmware backup!\n");
|
|
|
|
}
|
|
|
|
}
|
2017-06-25 22:35:19 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
fclose(f);
|
|
|
|
}
|
2021-10-02 18:16:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void LoadUserSettingsFromConfig()
|
|
|
|
{
|
|
|
|
// setting up username
|
2021-11-17 20:44:49 +00:00
|
|
|
std::string orig_username = Platform::GetConfigString(Platform::Firm_Username);
|
|
|
|
std::u16string username = std::wstring_convert<std::codecvt_utf8_utf16<char16_t>, char16_t>{}.from_bytes(orig_username);
|
2021-10-02 18:16:27 +00:00
|
|
|
size_t usernameLength = std::min(username.length(), (size_t) 10);
|
|
|
|
memcpy(Firmware + UserSettings + 0x06, username.data(), usernameLength * sizeof(char16_t));
|
|
|
|
Firmware[UserSettings+0x1A] = usernameLength;
|
|
|
|
|
|
|
|
// setting language
|
2021-11-17 20:44:49 +00:00
|
|
|
Firmware[UserSettings+0x64] = Platform::GetConfigInt(Platform::Firm_Language);
|
2021-10-02 18:16:27 +00:00
|
|
|
|
|
|
|
// setting up color
|
2021-11-17 20:44:49 +00:00
|
|
|
Firmware[UserSettings+0x02] = Platform::GetConfigInt(Platform::Firm_Color);
|
2021-10-02 18:16:27 +00:00
|
|
|
|
|
|
|
// setting up birthday
|
2021-11-17 20:44:49 +00:00
|
|
|
Firmware[UserSettings+0x03] = Platform::GetConfigInt(Platform::Firm_BirthdayMonth);
|
|
|
|
Firmware[UserSettings+0x04] = Platform::GetConfigInt(Platform::Firm_BirthdayDay);
|
2021-10-02 18:16:27 +00:00
|
|
|
|
|
|
|
// setup message
|
2021-11-17 20:44:49 +00:00
|
|
|
std::string orig_message = Platform::GetConfigString(Platform::Firm_Message);
|
|
|
|
std::u16string message = std::wstring_convert<std::codecvt_utf8_utf16<char16_t>, char16_t>{}.from_bytes(orig_message);
|
2021-10-02 18:16:27 +00:00
|
|
|
size_t messageLength = std::min(message.length(), (size_t) 26);
|
|
|
|
memcpy(Firmware + UserSettings + 0x1C, message.data(), messageLength * sizeof(char16_t));
|
|
|
|
Firmware[UserSettings+0x50] = messageLength;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Reset()
|
|
|
|
{
|
|
|
|
if (Firmware) delete[] Firmware;
|
|
|
|
Firmware = NULL;
|
|
|
|
|
|
|
|
if (NDS::ConsoleType == 1)
|
2021-11-17 19:42:11 +00:00
|
|
|
FirmwarePath = Platform::GetConfigString(Platform::DSi_FirmwarePath);
|
2021-10-02 18:16:27 +00:00
|
|
|
else
|
2021-11-17 19:42:11 +00:00
|
|
|
FirmwarePath = Platform::GetConfigString(Platform::FirmwarePath);
|
2021-10-02 18:16:27 +00:00
|
|
|
|
|
|
|
FILE* f = Platform::OpenLocalFile(FirmwarePath, "rb");
|
|
|
|
if (!f)
|
|
|
|
{
|
|
|
|
printf("Firmware not found! Generating default firmware.\n");
|
|
|
|
LoadDefaultFirmware();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LoadFirmwareFromFile(f);
|
|
|
|
}
|
2017-06-25 22:35:19 +00:00
|
|
|
|
2017-03-20 23:53:04 +00:00
|
|
|
FirmwareMask = FirmwareLength - 1;
|
|
|
|
|
|
|
|
u32 userdata = 0x7FE00 & FirmwareMask;
|
|
|
|
if (*(u16*)&Firmware[userdata+0x170] == ((*(u16*)&Firmware[userdata+0x70] + 1) & 0x7F))
|
2017-01-31 23:24:36 +00:00
|
|
|
{
|
2017-03-20 23:53:04 +00:00
|
|
|
if (VerifyCRC16(0xFFFF, userdata+0x100, 0x70, userdata+0x172))
|
|
|
|
userdata += 0x100;
|
2017-01-31 23:24:36 +00:00
|
|
|
}
|
|
|
|
|
2017-03-20 23:53:04 +00:00
|
|
|
UserSettings = userdata;
|
|
|
|
|
2021-11-17 20:44:49 +00:00
|
|
|
if (!f || Platform::GetConfigBool(Platform::Firm_OverrideSettings))
|
2021-10-02 18:16:27 +00:00
|
|
|
LoadUserSettingsFromConfig();
|
2021-11-17 19:42:11 +00:00
|
|
|
|
2021-08-30 18:26:49 +00:00
|
|
|
// fix touchscreen coords
|
|
|
|
*(u16*)&Firmware[userdata+0x58] = 0;
|
|
|
|
*(u16*)&Firmware[userdata+0x5A] = 0;
|
|
|
|
Firmware[userdata+0x5C] = 0;
|
|
|
|
Firmware[userdata+0x5D] = 0;
|
|
|
|
*(u16*)&Firmware[userdata+0x5E] = 255<<4;
|
|
|
|
*(u16*)&Firmware[userdata+0x60] = 191<<4;
|
|
|
|
Firmware[userdata+0x62] = 255;
|
|
|
|
Firmware[userdata+0x63] = 191;
|
|
|
|
|
|
|
|
// disable autoboot
|
|
|
|
//Firmware[userdata+0x64] &= 0xBF;
|
|
|
|
|
|
|
|
*(u16*)&Firmware[userdata+0x72] = CRC16(&Firmware[userdata], 0x70, 0xFFFF);
|
|
|
|
|
2021-11-17 20:44:49 +00:00
|
|
|
if (Platform::GetConfigBool(Platform::Firm_RandomizeMAC))
|
2020-06-21 09:02:42 +00:00
|
|
|
{
|
2021-08-30 18:26:49 +00:00
|
|
|
// replace MAC address with random address
|
|
|
|
Firmware[0x36] = 0x00;
|
|
|
|
Firmware[0x37] = 0x09;
|
|
|
|
Firmware[0x38] = 0xBF;
|
|
|
|
Firmware[0x39] = rand()&0xFF;
|
|
|
|
Firmware[0x3A] = rand()&0xFF;
|
|
|
|
Firmware[0x3B] = rand()&0xFF;
|
|
|
|
|
|
|
|
*(u16*)&Firmware[0x2A] = CRC16(&Firmware[0x2C], *(u16*)&Firmware[0x2C], 0x0000);
|
2020-06-21 09:02:42 +00:00
|
|
|
}
|
2017-01-30 17:36:11 +00:00
|
|
|
|
2017-05-11 21:43:57 +00:00
|
|
|
printf("MAC: %02X:%02X:%02X:%02X:%02X:%02X\n",
|
|
|
|
Firmware[0x36], Firmware[0x37], Firmware[0x38],
|
|
|
|
Firmware[0x39], Firmware[0x3A], Firmware[0x3B]);
|
|
|
|
|
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");
|
2017-03-20 23:53:04 +00:00
|
|
|
printf("FW: AP1 CRC16 = %s\n", VerifyCRC16(0x0000, 0x7FA00&FirmwareMask, 0xFE, 0x7FAFE&FirmwareMask)?"GOOD":"BAD");
|
|
|
|
printf("FW: AP2 CRC16 = %s\n", VerifyCRC16(0x0000, 0x7FB00&FirmwareMask, 0xFE, 0x7FBFE&FirmwareMask)?"GOOD":"BAD");
|
|
|
|
printf("FW: AP3 CRC16 = %s\n", VerifyCRC16(0x0000, 0x7FC00&FirmwareMask, 0xFE, 0x7FCFE&FirmwareMask)?"GOOD":"BAD");
|
|
|
|
printf("FW: USER0 CRC16 = %s\n", VerifyCRC16(0xFFFF, 0x7FE00&FirmwareMask, 0x70, 0x7FE72&FirmwareMask)?"GOOD":"BAD");
|
|
|
|
printf("FW: USER1 CRC16 = %s\n", VerifyCRC16(0xFFFF, 0x7FF00&FirmwareMask, 0x70, 0x7FF72&FirmwareMask)?"GOOD":"BAD");
|
2017-01-16 03:47:37 +00:00
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2018-10-18 00:54:48 +00:00
|
|
|
void DoSavestate(Savestate* file)
|
|
|
|
{
|
|
|
|
file->Section("SPFW");
|
|
|
|
|
|
|
|
// CHECKME/TODO: trust the firmware to stay the same?????
|
|
|
|
// embedding the whole firmware in the savestate would be derpo tho??
|
|
|
|
|
|
|
|
file->Var32(&Hold);
|
|
|
|
file->Var8(&CurCmd);
|
|
|
|
file->Var32(&DataPos);
|
|
|
|
file->Var8(&Data);
|
|
|
|
|
|
|
|
file->Var8(&StatusReg);
|
|
|
|
file->Var32(&Addr);
|
|
|
|
}
|
|
|
|
|
2021-08-30 18:26:49 +00:00
|
|
|
void SetupDirectBoot(bool dsi)
|
2017-03-20 23:53:04 +00:00
|
|
|
{
|
2021-08-30 18:26:49 +00:00
|
|
|
if (dsi)
|
|
|
|
{
|
|
|
|
for (u32 i = 0; i < 6; i += 2)
|
|
|
|
DSi::ARM9Write16(0x02FFFCF4, *(u16*)&Firmware[0x36+i]); // MAC address
|
2017-03-20 23:53:04 +00:00
|
|
|
|
2021-08-30 18:26:49 +00:00
|
|
|
// checkme
|
|
|
|
DSi::ARM9Write16(0x02FFFCFA, *(u16*)&Firmware[0x3C]); // enabled channels
|
2017-03-20 23:53:04 +00:00
|
|
|
|
2021-08-30 18:26:49 +00:00
|
|
|
for (u32 i = 0; i < 0x70; i += 4)
|
|
|
|
DSi::ARM9Write32(0x02FFFC80+i, *(u32*)&Firmware[UserSettings+i]);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
NDS::ARM9Write32(0x027FF864, 0);
|
|
|
|
NDS::ARM9Write32(0x027FF868, *(u16*)&Firmware[0x20] << 3); // user settings offset
|
|
|
|
|
|
|
|
NDS::ARM9Write16(0x027FF874, *(u16*)&Firmware[0x26]); // CRC16 for data/gfx
|
|
|
|
NDS::ARM9Write16(0x027FF876, *(u16*)&Firmware[0x04]); // CRC16 for GUI/wifi code
|
|
|
|
|
|
|
|
for (u32 i = 0; i < 0x70; i += 4)
|
|
|
|
NDS::ARM9Write32(0x027FFC80+i, *(u32*)&Firmware[UserSettings+i]);
|
|
|
|
}
|
2017-03-20 23:53:04 +00:00
|
|
|
}
|
|
|
|
|
2017-04-07 15:37:49 +00:00
|
|
|
u8 GetConsoleType() { return Firmware[0x1D]; }
|
|
|
|
u8 GetWifiVersion() { return Firmware[0x2F]; }
|
2020-08-19 15:53:31 +00:00
|
|
|
u8 GetNWifiVersion() { return Firmware[0x1FD]; } // for DSi; will return 0xFF on a DS
|
2017-04-07 15:37:49 +00:00
|
|
|
u8 GetRFVersion() { return Firmware[0x40]; }
|
2019-08-04 00:16:16 +00:00
|
|
|
u8* GetWifiMAC() { return &Firmware[0x36]; }
|
2017-04-07 15:37:49 +00:00
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
u8 Read()
|
|
|
|
{
|
|
|
|
return Data;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Write(u8 val, u32 hold)
|
|
|
|
{
|
|
|
|
if (!hold)
|
|
|
|
{
|
2017-06-25 22:35:19 +00:00
|
|
|
if (!Hold) // commands with no paramters
|
|
|
|
CurCmd = val;
|
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
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;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (CurCmd)
|
|
|
|
{
|
|
|
|
case 0x03: // read
|
|
|
|
{
|
|
|
|
if (DataPos < 4)
|
|
|
|
{
|
|
|
|
Addr <<= 8;
|
|
|
|
Addr |= val;
|
|
|
|
Data = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-03-20 23:53:04 +00:00
|
|
|
Data = Firmware[Addr & FirmwareMask];
|
2016-12-04 02:20:50 +00:00
|
|
|
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;
|
|
|
|
|
2017-06-25 22:35:19 +00:00
|
|
|
case 0x0A: // write
|
|
|
|
{
|
|
|
|
// TODO: what happens if you write too many bytes? (max 256, they say)
|
|
|
|
if (DataPos < 4)
|
|
|
|
{
|
|
|
|
Addr <<= 8;
|
|
|
|
Addr |= val;
|
|
|
|
Data = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Firmware[Addr & FirmwareMask] = val;
|
|
|
|
Data = val;
|
|
|
|
Addr++;
|
|
|
|
}
|
|
|
|
|
|
|
|
DataPos++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2016-12-05 17:02:29 +00:00
|
|
|
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-06-25 22:35:19 +00:00
|
|
|
|
|
|
|
if (!hold && (CurCmd == 0x02 || CurCmd == 0x0A))
|
|
|
|
{
|
2020-05-17 02:37:44 +00:00
|
|
|
FILE* f = Platform::OpenLocalFile(FirmwarePath, "r+b");
|
2017-06-25 22:35:19 +00:00
|
|
|
if (f)
|
|
|
|
{
|
|
|
|
u32 cutoff = 0x7FA00 & FirmwareMask;
|
|
|
|
fseek(f, cutoff, SEEK_SET);
|
|
|
|
fwrite(&Firmware[cutoff], FirmwareLength-cutoff, 1, f);
|
|
|
|
fclose(f);
|
|
|
|
}
|
|
|
|
}
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
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];
|
|
|
|
|
|
|
|
|
2017-02-07 21:23:46 +00:00
|
|
|
bool Init()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeInit()
|
2017-01-20 00:18:30 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2018-10-18 00:54:48 +00:00
|
|
|
void DoSavestate(Savestate* file)
|
|
|
|
{
|
|
|
|
file->Section("SPPW");
|
|
|
|
|
|
|
|
file->Var32(&Hold);
|
|
|
|
file->Var32(&DataPos);
|
|
|
|
file->Var8(&Index);
|
|
|
|
file->Var8(&Data);
|
|
|
|
|
|
|
|
file->VarArray(Registers, 8);
|
|
|
|
file->VarArray(RegMasks, 8); // is that needed??
|
|
|
|
}
|
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
u8 Read()
|
|
|
|
{
|
|
|
|
return Data;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Write(u8 val, u32 hold)
|
2017-09-21 01:59:12 +00:00
|
|
|
{
|
2017-01-20 00:18:30 +00:00
|
|
|
if (!hold)
|
|
|
|
{
|
|
|
|
Hold = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hold && (!Hold))
|
|
|
|
{
|
|
|
|
Index = val;
|
|
|
|
Hold = 1;
|
|
|
|
Data = 0;
|
|
|
|
DataPos = 1;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (DataPos == 1)
|
|
|
|
{
|
2017-08-05 17:13:55 +00:00
|
|
|
u32 regid = Index & 0x07;
|
|
|
|
|
2017-01-20 00:18:30 +00:00
|
|
|
if (Index & 0x80)
|
|
|
|
{
|
2017-08-05 17:13:55 +00:00
|
|
|
Data = Registers[regid];
|
2017-01-20 00:18:30 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-08-05 17:13:55 +00:00
|
|
|
Registers[regid] = (Registers[regid] & ~RegMasks[regid]) | (val & RegMasks[regid]);
|
|
|
|
|
|
|
|
switch (regid)
|
|
|
|
{
|
|
|
|
case 0:
|
2017-09-21 01:59:12 +00:00
|
|
|
if (val & 0x40) NDS::Stop(); // shutdown
|
|
|
|
//printf("power %02X\n", val);
|
2017-08-05 17:13:55 +00:00
|
|
|
break;
|
|
|
|
case 4:
|
2017-09-21 01:59:12 +00:00
|
|
|
//printf("brightness %02X\n", val);
|
2017-08-05 17:13:55 +00:00
|
|
|
break;
|
|
|
|
}
|
2017-01-20 00:18:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
Data = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
|
2017-01-31 23:24:36 +00:00
|
|
|
namespace SPI_TSC
|
|
|
|
{
|
|
|
|
|
|
|
|
u32 DataPos;
|
|
|
|
u8 ControlByte;
|
|
|
|
u8 Data;
|
|
|
|
|
|
|
|
u16 ConvResult;
|
|
|
|
|
|
|
|
u16 TouchX, TouchY;
|
|
|
|
|
2018-12-12 15:33:40 +00:00
|
|
|
s16 MicBuffer[1024];
|
|
|
|
int MicBufferLen;
|
|
|
|
|
2017-01-31 23:24:36 +00:00
|
|
|
|
2017-02-07 21:23:46 +00:00
|
|
|
bool Init()
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeInit()
|
2017-01-31 23:24:36 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void Reset()
|
|
|
|
{
|
|
|
|
ControlByte = 0;
|
|
|
|
Data = 0;
|
|
|
|
|
|
|
|
ConvResult = 0;
|
2018-12-12 15:33:40 +00:00
|
|
|
|
|
|
|
MicBufferLen = 0;
|
2017-01-31 23:24:36 +00:00
|
|
|
}
|
|
|
|
|
2018-10-18 00:54:48 +00:00
|
|
|
void DoSavestate(Savestate* file)
|
|
|
|
{
|
|
|
|
file->Section("SPTS");
|
|
|
|
|
|
|
|
file->Var32(&DataPos);
|
|
|
|
file->Var8(&ControlByte);
|
|
|
|
file->Var8(&Data);
|
|
|
|
|
|
|
|
file->Var16(&ConvResult);
|
|
|
|
}
|
|
|
|
|
2017-01-31 23:24:36 +00:00
|
|
|
void SetTouchCoords(u16 x, u16 y)
|
|
|
|
{
|
|
|
|
// scr.x = (adc.x-adc.x1) * (scr.x2-scr.x1) / (adc.x2-adc.x1) + (scr.x1-1)
|
|
|
|
// scr.y = (adc.y-adc.y1) * (scr.y2-scr.y1) / (adc.y2-adc.y1) + (scr.y1-1)
|
|
|
|
// adc.x = ((scr.x * ((adc.x2-adc.x1) + (scr.x1-1))) / (scr.x2-scr.x1)) + adc.x1
|
|
|
|
// adc.y = ((scr.y * ((adc.y2-adc.y1) + (scr.y1-1))) / (scr.y2-scr.y1)) + adc.y1
|
|
|
|
TouchX = x;
|
|
|
|
TouchY = y;
|
|
|
|
|
|
|
|
if (y == 0xFFF) return;
|
|
|
|
|
2017-02-03 23:12:08 +00:00
|
|
|
TouchX <<= 4;
|
|
|
|
TouchY <<= 4;
|
2017-01-31 23:24:36 +00:00
|
|
|
}
|
|
|
|
|
2018-12-12 15:33:40 +00:00
|
|
|
void MicInputFrame(s16* data, int samples)
|
|
|
|
{
|
2018-12-14 04:15:57 +00:00
|
|
|
if (!data)
|
|
|
|
{
|
|
|
|
MicBufferLen = 0;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-12-12 15:33:40 +00:00
|
|
|
if (samples > 1024) samples = 1024;
|
|
|
|
memcpy(MicBuffer, data, samples*sizeof(s16));
|
|
|
|
MicBufferLen = samples;
|
|
|
|
}
|
|
|
|
|
2017-01-31 23:24:36 +00:00
|
|
|
u8 Read()
|
|
|
|
{
|
|
|
|
return Data;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Write(u8 val, u32 hold)
|
|
|
|
{
|
|
|
|
if (DataPos == 1)
|
|
|
|
Data = (ConvResult >> 5) & 0xFF;
|
|
|
|
else if (DataPos == 2)
|
|
|
|
Data = (ConvResult << 3) & 0xFF;
|
|
|
|
else
|
|
|
|
Data = 0;
|
|
|
|
|
|
|
|
if (val & 0x80)
|
|
|
|
{
|
|
|
|
ControlByte = val;
|
|
|
|
DataPos = 1;
|
|
|
|
|
|
|
|
switch (ControlByte & 0x70)
|
|
|
|
{
|
|
|
|
case 0x10: ConvResult = TouchY; break;
|
|
|
|
case 0x50: ConvResult = TouchX; break;
|
2018-12-12 15:33:40 +00:00
|
|
|
|
|
|
|
case 0x60:
|
|
|
|
{
|
|
|
|
if (MicBufferLen == 0)
|
|
|
|
ConvResult = 0x800;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// 560190 cycles per frame
|
|
|
|
u32 cyclepos = (u32)NDS::GetSysClockCycles(2);
|
|
|
|
u32 samplepos = (cyclepos * MicBufferLen) / 560190;
|
2018-12-12 16:23:54 +00:00
|
|
|
if (samplepos >= MicBufferLen) samplepos = MicBufferLen-1;
|
2018-12-12 15:33:40 +00:00
|
|
|
s16 sample = MicBuffer[samplepos];
|
|
|
|
|
|
|
|
// make it louder
|
|
|
|
//if (sample > 0x3FFF) sample = 0x7FFF;
|
|
|
|
//else if (sample < -0x4000) sample = -0x8000;
|
|
|
|
//else sample <<= 1;
|
|
|
|
|
|
|
|
// make it unsigned 12-bit
|
|
|
|
sample ^= 0x8000;
|
|
|
|
ConvResult = sample >> 4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2017-01-31 23:24:36 +00:00
|
|
|
default: ConvResult = 0xFFF; break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ControlByte & 0x08)
|
|
|
|
ConvResult &= 0x0FF0; // checkme
|
|
|
|
}
|
|
|
|
else
|
|
|
|
DataPos++;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
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
|
|
|
|
2019-06-15 14:58:02 +00:00
|
|
|
u32 CurDevice; // remove me
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
|
2017-02-07 21:23:46 +00:00
|
|
|
bool Init()
|
2016-12-04 02:20:50 +00:00
|
|
|
{
|
2017-02-07 21:23:46 +00:00
|
|
|
if (!SPI_Firmware::Init()) return false;
|
|
|
|
if (!SPI_Powerman::Init()) return false;
|
|
|
|
if (!SPI_TSC::Init()) return false;
|
2019-08-04 09:44:36 +00:00
|
|
|
if (!DSi_SPI_TSC::Init()) return false;
|
2017-02-07 21:23:46 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeInit()
|
|
|
|
{
|
|
|
|
SPI_Firmware::DeInit();
|
|
|
|
SPI_Powerman::DeInit();
|
|
|
|
SPI_TSC::DeInit();
|
2019-08-04 09:44:36 +00:00
|
|
|
DSi_SPI_TSC::DeInit();
|
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();
|
2017-05-25 20:46:01 +00:00
|
|
|
SPI_TSC::Reset();
|
2020-06-01 18:36:30 +00:00
|
|
|
if (NDS::ConsoleType == 1) DSi_SPI_TSC::Reset();
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
|
2018-10-18 00:54:48 +00:00
|
|
|
void DoSavestate(Savestate* file)
|
|
|
|
{
|
|
|
|
file->Section("SPIG");
|
|
|
|
|
|
|
|
file->Var16(&Cnt);
|
|
|
|
file->Var32(&CurDevice);
|
|
|
|
|
|
|
|
SPI_Firmware::DoSavestate(file);
|
|
|
|
SPI_Powerman::DoSavestate(file);
|
|
|
|
SPI_TSC::DoSavestate(file);
|
2020-06-01 18:36:30 +00:00
|
|
|
if (NDS::ConsoleType == 1) DSi_SPI_TSC::DoSavestate(file);
|
2018-10-18 00:54:48 +00:00
|
|
|
}
|
|
|
|
|
2016-12-04 02:20:50 +00:00
|
|
|
|
|
|
|
void WriteCnt(u16 val)
|
|
|
|
{
|
2017-12-18 23:20:36 +00:00
|
|
|
// turning it off should clear chipselect
|
|
|
|
// TODO: confirm on hardware. libnds expects this, though.
|
|
|
|
if ((Cnt & (1<<15)) && !(val & (1<<15)))
|
|
|
|
{
|
|
|
|
switch (Cnt & 0x0300)
|
|
|
|
{
|
|
|
|
case 0x0000: SPI_Powerman::Hold = 0; break;
|
|
|
|
case 0x0100: SPI_Firmware::Hold = 0; break;
|
2020-06-01 18:36:30 +00:00
|
|
|
case 0x0200:
|
|
|
|
if (NDS::ConsoleType == 1)
|
|
|
|
DSi_SPI_TSC::DataPos = 0;
|
|
|
|
else
|
|
|
|
SPI_TSC::DataPos = 0;
|
|
|
|
break;
|
2017-12-18 23:20:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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");
|
2017-07-15 17:29:10 +00:00
|
|
|
if (Cnt & (1<<7)) printf("!! CHANGING SPICNT DURING TRANSFER: %04X\n", val);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TransferDone(u32 param)
|
|
|
|
{
|
|
|
|
Cnt &= ~(1<<7);
|
|
|
|
|
|
|
|
if (Cnt & (1<<14))
|
|
|
|
NDS::SetIRQ(1, NDS::IRQ_SPI);
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
u8 ReadData()
|
|
|
|
{
|
2017-01-31 16:34:17 +00:00
|
|
|
if (!(Cnt & (1<<15))) return 0;
|
2017-07-15 17:29:10 +00:00
|
|
|
if (Cnt & (1<<7)) return 0; // checkme
|
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();
|
2020-06-01 18:36:30 +00:00
|
|
|
case 0x0200:
|
|
|
|
if (NDS::ConsoleType == 1)
|
|
|
|
return DSi_SPI_TSC::Read();
|
|
|
|
else
|
|
|
|
return SPI_TSC::Read();
|
2016-12-04 02:20:50 +00:00
|
|
|
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
|
|
|
|
2017-07-15 17:29:10 +00:00
|
|
|
if (Cnt & (1<<7)) printf("!! WRITING AUXSPIDATA DURING PENDING TRANSFER\n");
|
2016-12-04 02:20:50 +00:00
|
|
|
|
2017-07-15 17:29:10 +00:00
|
|
|
Cnt |= (1<<7);
|
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;
|
2020-06-01 18:36:30 +00:00
|
|
|
case 0x0200:
|
|
|
|
if (NDS::ConsoleType == 1)
|
|
|
|
DSi_SPI_TSC::Write(val, Cnt&(1<<11));
|
|
|
|
else
|
|
|
|
SPI_TSC::Write(val, Cnt&(1<<11));
|
|
|
|
break;
|
2017-01-31 16:34:17 +00:00
|
|
|
default: printf("SPI to unknown device %04X %02X\n", Cnt, val); break;
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
|
2017-07-15 17:29:10 +00:00
|
|
|
// SPI transfers one bit per cycle -> 8 cycles per byte
|
|
|
|
u32 delay = 8 * (8 << (Cnt & 0x3));
|
|
|
|
NDS::ScheduleEvent(NDS::Event_SPITransfer, false, delay, TransferDone, 0);
|
2016-12-04 02:20:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|