2023-12-14 19:26:54 +01:00
|
|
|
// Original code sourced from https://github.com/jago85/UltraCIC_C
|
|
|
|
|
|
|
|
// MIT License
|
|
|
|
|
|
|
|
// Copyright (c) 2019 Jan Goldacker
|
2024-04-10 12:56:05 +02:00
|
|
|
// Copyright (c) 2022-2024 Mateusz Faderewski
|
2023-12-14 19:26:54 +01:00
|
|
|
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
|
|
// in the Software without restriction, including without limitation the rights
|
|
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
|
|
// furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
// The above copyright notice and this permission notice shall be included in all
|
|
|
|
// copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
// SOFTWARE.
|
|
|
|
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
volatile uint32_t CIC_CONFIG[2];
|
2024-04-10 12:56:05 +02:00
|
|
|
volatile uint32_t IO;
|
|
|
|
volatile uint32_t DEBUG;
|
2023-12-14 19:26:54 +01:00
|
|
|
} ext_regs_t;
|
|
|
|
|
|
|
|
#define EXT ((ext_regs_t *) (0xC0000000UL))
|
|
|
|
|
|
|
|
#define CIC_DQ (1 << 0)
|
|
|
|
#define CIC_CLK (1 << 1)
|
|
|
|
#define CIC_RESET (1 << 2)
|
|
|
|
#define CIC_INVALID_REGION (1 << 3)
|
2024-04-10 12:56:05 +02:00
|
|
|
#define CIC_TIMER_RESET (1 << 4)
|
|
|
|
#define CIC_TIMER_BIT (16)
|
|
|
|
#define CIC_TIMER_MASK (0xFFF)
|
|
|
|
|
|
|
|
#define CIC_IS_RUNNING() (EXT->IO & CIC_RESET)
|
|
|
|
#define CIC_CLK_WAIT_LOW() { while ((EXT->IO & (CIC_RESET | CIC_CLK)) == (CIC_RESET | CIC_CLK)); }
|
|
|
|
#define CIC_CLK_WAIT_HIGH() { while ((EXT->IO & (CIC_RESET | CIC_CLK)) == CIC_RESET); }
|
|
|
|
#define CIC_DQ_GET() (EXT->IO & CIC_DQ)
|
|
|
|
#define CIC_DQ_SET(v) { EXT->IO = ((v) ? CIC_DQ : 0); }
|
|
|
|
#define CIC_CLK_GET() (EXT->IO & (CIC_RESET | CIC_CLK))
|
|
|
|
#define CIC_NOTIFY_INVALID_REGION() { EXT->IO = (CIC_INVALID_REGION | CIC_DQ); }
|
|
|
|
#define CIC_TIMEOUT_START() { EXT->IO = (CIC_TIMER_RESET | CIC_DQ); }
|
|
|
|
#define CIC_TIMEOUT_GET() ((EXT->IO >> CIC_TIMER_BIT) & CIC_TIMER_MASK)
|
|
|
|
|
|
|
|
#define CIC_TIMER_MS_TO_TICKS(ms) ((((62500000 / 32) / 256) * (ms)) / 1000)
|
|
|
|
|
|
|
|
#define CIC_TIMEOUT_INVALID_REGION CIC_TIMER_MS_TO_TICKS(100)
|
|
|
|
#define CIC_TIMEOUT_SOFT_RESET CIC_TIMER_MS_TO_TICKS(500)
|
|
|
|
|
|
|
|
typedef enum {
|
|
|
|
CIC_STEP_UNINITIALIZED = 0,
|
|
|
|
CIC_STEP_POWER_OFF = 1,
|
|
|
|
CIC_STEP_INIT = 2,
|
|
|
|
CIC_STEP_ID = 3,
|
|
|
|
CIC_STEP_SEED = 4,
|
|
|
|
CIC_STEP_CHECKSUM = 5,
|
|
|
|
CIC_STEP_INIT_RAM = 6,
|
|
|
|
CIC_STEP_COMMAND = 7,
|
|
|
|
CIC_STEP_COMPARE = 8,
|
|
|
|
CIC_STEP_X105 = 9,
|
|
|
|
CIC_STEP_RESET_BUTTON = 10,
|
|
|
|
CIC_STEP_DIE_DISABLED = 11,
|
|
|
|
CIC_STEP_DIE_64DD = 12,
|
|
|
|
CIC_STEP_DIE_INVALID_REGION = 13,
|
|
|
|
CIC_STEP_DIE_COMMAND = 14,
|
|
|
|
} cic_step_t;
|
2023-12-14 19:26:54 +01:00
|
|
|
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
bool cic_disabled;
|
|
|
|
bool cic_64dd_mode;
|
|
|
|
bool cic_region;
|
|
|
|
uint8_t cic_seed;
|
|
|
|
uint8_t cic_checksum[6];
|
|
|
|
} cic_config_t;
|
|
|
|
|
|
|
|
static cic_config_t config;
|
|
|
|
|
2024-04-10 12:56:05 +02:00
|
|
|
static uint16_t timeout;
|
|
|
|
|
2023-12-14 19:26:54 +01:00
|
|
|
static uint8_t cic_ram[32];
|
|
|
|
static uint8_t cic_x105_ram[30];
|
|
|
|
|
2024-04-10 12:56:05 +02:00
|
|
|
static const uint8_t cic_ram_init[2][16] = {{
|
|
|
|
0xE0, 0x9A, 0x18, 0x5A, 0x13, 0xE1, 0x0D, 0xEC,
|
|
|
|
0x0B, 0x14, 0xF8, 0xB5, 0x7C, 0xD6, 0x1E, 0x98
|
2023-12-14 19:26:54 +01:00
|
|
|
}, {
|
2024-04-10 12:56:05 +02:00
|
|
|
0xE0, 0x4F, 0x51, 0x21, 0x71, 0x98, 0x57, 0x5A,
|
|
|
|
0x0B, 0x12, 0x3F, 0x82, 0x71, 0x98, 0x11, 0x5C
|
2023-12-14 19:26:54 +01:00
|
|
|
}};
|
|
|
|
|
|
|
|
|
2024-04-10 12:56:05 +02:00
|
|
|
static void cic_set_step (cic_step_t step) {
|
|
|
|
EXT->DEBUG = step;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_timeout_start (uint16_t timeout_ticks) {
|
|
|
|
CIC_TIMEOUT_START();
|
|
|
|
timeout = timeout_ticks;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool cic_timeout_elapsed (void) {
|
|
|
|
return (CIC_TIMEOUT_GET() >= timeout);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_die (cic_step_t reason) {
|
|
|
|
cic_set_step(reason);
|
2023-12-14 19:26:54 +01:00
|
|
|
while (CIC_IS_RUNNING());
|
|
|
|
}
|
|
|
|
|
2024-04-10 12:56:05 +02:00
|
|
|
static void cic_wait_power_on (void) {
|
2023-12-14 19:26:54 +01:00
|
|
|
CIC_DQ_SET(1);
|
|
|
|
while (!CIC_IS_RUNNING());
|
2024-04-10 12:56:05 +02:00
|
|
|
}
|
2023-12-14 19:26:54 +01:00
|
|
|
|
2024-04-10 12:56:05 +02:00
|
|
|
static void cic_init (void) {
|
2023-12-14 19:26:54 +01:00
|
|
|
uint32_t cic_config[2];
|
|
|
|
|
|
|
|
cic_config[0] = EXT->CIC_CONFIG[0];
|
|
|
|
cic_config[1] = EXT->CIC_CONFIG[1];
|
|
|
|
|
|
|
|
config.cic_disabled = (cic_config[0] & (1 << 26));
|
|
|
|
config.cic_64dd_mode = (cic_config[0] & (1 << 25));
|
|
|
|
config.cic_region = (cic_config[0] & (1 << 24));
|
|
|
|
config.cic_seed = ((cic_config[0] >> 16) & 0xFF);
|
|
|
|
config.cic_checksum[0] = ((cic_config[0] >> 8) & 0xFF);
|
|
|
|
config.cic_checksum[1] = (cic_config[0] & 0xFF);
|
|
|
|
config.cic_checksum[2] = ((cic_config[1] >> 24) & 0xFF);
|
|
|
|
config.cic_checksum[3] = ((cic_config[1] >> 16) & 0xFF);
|
|
|
|
config.cic_checksum[4] = ((cic_config[1] >> 8) & 0xFF);
|
|
|
|
config.cic_checksum[5] = (cic_config[1] & 0xFF);
|
|
|
|
|
|
|
|
if (config.cic_disabled) {
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_die(CIC_STEP_DIE_DISABLED);
|
2023-12-14 19:26:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint8_t cic_read (void) {
|
|
|
|
uint8_t value;
|
|
|
|
CIC_CLK_WAIT_LOW();
|
|
|
|
value = CIC_DQ_GET() ? 1 : 0;
|
|
|
|
CIC_CLK_WAIT_HIGH();
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_write (uint8_t value) {
|
|
|
|
CIC_CLK_WAIT_LOW();
|
|
|
|
CIC_DQ_SET(value);
|
|
|
|
CIC_CLK_WAIT_HIGH();
|
|
|
|
CIC_DQ_SET(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint8_t cic_read_nibble (void) {
|
|
|
|
uint8_t data = 0;
|
|
|
|
for (int i = 0; i < 4; i++) {
|
|
|
|
data = ((data << 1) | cic_read());
|
|
|
|
}
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_write_nibble (uint8_t data) {
|
|
|
|
cic_write(data & 0x08);
|
|
|
|
cic_write(data & 0x04);
|
|
|
|
cic_write(data & 0x02);
|
|
|
|
cic_write(data & 0x01);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_write_ram_nibbles (uint8_t index) {
|
|
|
|
do {
|
|
|
|
cic_write_nibble(cic_ram[index++]);
|
|
|
|
} while ((index & 0x0F) != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_encode_round (uint8_t index) {
|
|
|
|
uint8_t data = cic_ram[index++];
|
|
|
|
do {
|
|
|
|
data = ((((data + 1) & 0x0F) + cic_ram[index]) & 0x0F);
|
|
|
|
cic_ram[index++] = data;
|
|
|
|
} while ((index & 0x0F) != 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_write_id (void) {
|
|
|
|
if (config.cic_64dd_mode) {
|
|
|
|
CIC_CLK_WAIT_LOW();
|
|
|
|
while (CIC_CLK_GET() == CIC_RESET) {
|
|
|
|
if (!CIC_DQ_GET()) {
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_die(CIC_STEP_DIE_64DD);
|
2023-12-14 19:26:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
cic_write(0);
|
|
|
|
}
|
|
|
|
cic_write(config.cic_region ? 1 : 0);
|
|
|
|
cic_write(0);
|
|
|
|
cic_write(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_write_seed (void) {
|
|
|
|
cic_ram[0x0A] = 0x0B;
|
|
|
|
cic_ram[0x0B] = 0x05;
|
|
|
|
cic_ram[0x0C] = (config.cic_seed >> 4);
|
|
|
|
cic_ram[0x0D] = config.cic_seed;
|
|
|
|
cic_ram[0x0E] = (config.cic_seed >> 4);
|
|
|
|
cic_ram[0x0F] = config.cic_seed;
|
|
|
|
cic_encode_round(0x0A);
|
|
|
|
cic_encode_round(0x0A);
|
|
|
|
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_timeout_start(CIC_TIMEOUT_INVALID_REGION);
|
|
|
|
|
|
|
|
while (CIC_CLK_GET() == (CIC_RESET | CIC_CLK)) {
|
|
|
|
if (cic_timeout_elapsed()) {
|
2023-12-14 19:26:54 +01:00
|
|
|
CIC_NOTIFY_INVALID_REGION();
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_die(CIC_STEP_DIE_INVALID_REGION);
|
2023-12-14 19:26:54 +01:00
|
|
|
}
|
2024-04-10 12:56:05 +02:00
|
|
|
}
|
2023-12-14 19:26:54 +01:00
|
|
|
|
|
|
|
cic_write_ram_nibbles(0x0A);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_write_checksum (void) {
|
|
|
|
for (int i = 0; i < 4; i++) {
|
|
|
|
cic_ram[i] = 0x00;
|
|
|
|
}
|
|
|
|
for (int i = 0; i < 6; i++) {
|
|
|
|
cic_ram[(i * 2) + 4] = ((config.cic_checksum[i] >> 4) & 0x0F);
|
|
|
|
cic_ram[(i * 2) + 5] = (config.cic_checksum[i] & 0x0F);
|
|
|
|
}
|
|
|
|
cic_encode_round(0x00);
|
|
|
|
cic_encode_round(0x00);
|
|
|
|
cic_encode_round(0x00);
|
|
|
|
cic_encode_round(0x00);
|
|
|
|
cic_write(0);
|
|
|
|
cic_write_ram_nibbles(0x00);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_init_ram (void) {
|
2024-04-10 12:56:05 +02:00
|
|
|
for (int i = 0; i < 32; i += 2) {
|
|
|
|
uint8_t value = cic_ram_init[config.cic_region ? 1 : 0][i / 2];
|
|
|
|
cic_ram[i] = ((value >> 4) & 0x0F);
|
|
|
|
cic_ram[i + 1] = (value & 0x0F);
|
2023-12-14 19:26:54 +01:00
|
|
|
}
|
|
|
|
cic_ram[0x01] = cic_read_nibble();
|
|
|
|
cic_ram[0x11] = cic_read_nibble();
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_exchange_bytes (uint8_t *a, uint8_t *b) {
|
|
|
|
uint8_t tmp = *a;
|
|
|
|
*a = *b;
|
|
|
|
*b = tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_round (uint8_t *m) {
|
|
|
|
uint8_t a, b, x;
|
|
|
|
|
|
|
|
x = m[15];
|
|
|
|
a = x;
|
|
|
|
|
|
|
|
do {
|
|
|
|
b = 1;
|
|
|
|
a += (m[b] + 1);
|
|
|
|
m[b] = a;
|
|
|
|
b++;
|
|
|
|
a += (m[b] + 1);
|
|
|
|
cic_exchange_bytes(&a, &m[b]);
|
|
|
|
m[b] = ~(m[b]);
|
|
|
|
b++;
|
|
|
|
a &= 0x0F;
|
|
|
|
a += ((m[b] & 0x0F) + 1);
|
|
|
|
if (a < 16) {
|
|
|
|
cic_exchange_bytes(&a, &m[b]);
|
|
|
|
b++;
|
|
|
|
}
|
|
|
|
a += m[b];
|
|
|
|
m[b] = a;
|
|
|
|
b++;
|
|
|
|
a += m[b];
|
|
|
|
cic_exchange_bytes(&a, &m[b]);
|
|
|
|
b++;
|
|
|
|
a &= 0x0F;
|
|
|
|
a += 8;
|
|
|
|
if (a < 16) {
|
|
|
|
a += m[b];
|
|
|
|
}
|
|
|
|
cic_exchange_bytes(&a, &m[b]);
|
|
|
|
b++;
|
|
|
|
do {
|
|
|
|
a += (m[b] + 1);
|
|
|
|
m[b] = a;
|
|
|
|
b++;
|
|
|
|
b &= 0x0F;
|
|
|
|
} while (b != 0);
|
|
|
|
a = (x + 0x0F);
|
|
|
|
x = (a & 0x0F);
|
|
|
|
} while (x != 0x0F);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_compare_mode (void) {
|
|
|
|
cic_round(&cic_ram[0x10]);
|
|
|
|
cic_round(&cic_ram[0x10]);
|
|
|
|
cic_round(&cic_ram[0x10]);
|
|
|
|
|
|
|
|
uint8_t index = (cic_ram[0x17] & 0x0F);
|
|
|
|
if (index == 0) {
|
|
|
|
index = 1;
|
|
|
|
}
|
|
|
|
index |= 0x10;
|
|
|
|
|
|
|
|
do {
|
|
|
|
cic_read();
|
|
|
|
cic_write(cic_ram[index] & 0x01);
|
|
|
|
if (config.cic_region) {
|
|
|
|
index--;
|
|
|
|
} else {
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
} while (index & 0x0F);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_x105_algorithm (void) {
|
|
|
|
uint8_t a = 5;
|
|
|
|
uint8_t carry = 1;
|
|
|
|
|
|
|
|
for (int i = 0; i < 30; ++i) {
|
|
|
|
if (!(cic_x105_ram[i] & 0x01)) {
|
|
|
|
a += 8;
|
|
|
|
}
|
|
|
|
if (!(a & 0x02)) {
|
|
|
|
a += 4;
|
|
|
|
}
|
|
|
|
a = ((a + cic_x105_ram[i]) & 0x0F);
|
|
|
|
cic_x105_ram[i] = a;
|
|
|
|
if (!carry) {
|
|
|
|
a += 7;
|
|
|
|
}
|
|
|
|
a = ((a + cic_x105_ram[i]) & 0x0F);
|
|
|
|
a = (a + cic_x105_ram[i] + carry);
|
|
|
|
if (a >= 0x10) {
|
|
|
|
carry = 1;
|
|
|
|
a -= 0x10;
|
|
|
|
} else {
|
|
|
|
carry = 0;
|
|
|
|
}
|
|
|
|
a = (~(a) & 0x0F);
|
|
|
|
cic_x105_ram[i] = a;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_x105_mode (void) {
|
|
|
|
cic_write_nibble(0x0A);
|
|
|
|
cic_write_nibble(0x0A);
|
|
|
|
|
|
|
|
for (int i = 0; i < 30; i++) {
|
|
|
|
cic_x105_ram[i] = cic_read_nibble();
|
|
|
|
}
|
|
|
|
|
|
|
|
cic_x105_algorithm();
|
|
|
|
|
|
|
|
cic_write(0);
|
|
|
|
|
|
|
|
for (int i = 0; i < 30; i++) {
|
|
|
|
cic_write_nibble(cic_x105_ram[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cic_soft_reset (void) {
|
|
|
|
CIC_CLK_WAIT_LOW();
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_timeout_start(CIC_TIMEOUT_SOFT_RESET);
|
|
|
|
while ((!cic_timeout_elapsed()) && CIC_IS_RUNNING());
|
2023-12-14 19:26:54 +01:00
|
|
|
cic_write(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
__attribute__((naked)) void cic_main (void) {
|
|
|
|
while (true) {
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_set_step(CIC_STEP_POWER_OFF);
|
|
|
|
cic_wait_power_on();
|
|
|
|
|
|
|
|
cic_set_step(CIC_STEP_INIT);
|
2023-12-14 19:26:54 +01:00
|
|
|
cic_init();
|
|
|
|
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_set_step(CIC_STEP_ID);
|
2023-12-14 19:26:54 +01:00
|
|
|
cic_write_id();
|
2024-04-10 12:56:05 +02:00
|
|
|
|
|
|
|
cic_set_step(CIC_STEP_SEED);
|
2023-12-14 19:26:54 +01:00
|
|
|
cic_write_seed();
|
2024-04-10 12:56:05 +02:00
|
|
|
|
|
|
|
cic_set_step(CIC_STEP_CHECKSUM);
|
2023-12-14 19:26:54 +01:00
|
|
|
cic_write_checksum();
|
2024-04-10 12:56:05 +02:00
|
|
|
|
|
|
|
cic_set_step(CIC_STEP_INIT_RAM);
|
2023-12-14 19:26:54 +01:00
|
|
|
cic_init_ram();
|
|
|
|
|
|
|
|
while (CIC_IS_RUNNING()) {
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_set_step(CIC_STEP_COMMAND);
|
2023-12-14 19:26:54 +01:00
|
|
|
uint8_t cmd = 0;
|
|
|
|
cmd |= (cic_read() << 1);
|
|
|
|
cmd |= cic_read();
|
|
|
|
|
|
|
|
if (cmd == 0) {
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_set_step(CIC_STEP_COMPARE);
|
2023-12-14 19:26:54 +01:00
|
|
|
cic_compare_mode();
|
|
|
|
} else if (cmd == 2) {
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_set_step(CIC_STEP_X105);
|
2023-12-14 19:26:54 +01:00
|
|
|
cic_x105_mode();
|
|
|
|
} else if (cmd == 3) {
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_set_step(CIC_STEP_RESET_BUTTON);
|
2023-12-14 19:26:54 +01:00
|
|
|
cic_soft_reset();
|
|
|
|
} else {
|
2024-04-10 12:56:05 +02:00
|
|
|
cic_die(CIC_STEP_DIE_COMMAND);
|
2023-12-14 19:26:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|