implemented ultracic detection code

This commit is contained in:
Robin Jones 2017-11-10 15:56:02 +00:00
parent b7dd066fa0
commit 2034bedc1a
2 changed files with 263 additions and 0 deletions

166
inc/regsinternal.h Normal file
View File

@ -0,0 +1,166 @@
//TODO: this file is actually part of the libdragon source and should be included from there!!!
/**
* @file regsinternal.h
* @brief Register definitions for various hardware in the N64
* @ingroup lowlevel
*/
#ifndef __LIBDRAGON_REGSINTERNAL_H
#define __LIBDRAGON_REGSINTERNAL_H
/**
* @defgroup lowlevel Low Level Hardware Interfaces
* @ingroup libdragon
* @brief Low level hardware interface descriptions and functionality
*
* The low level hardware interfaces handle several functions in the N64 that
* would otherwise be handled by a kernel or RTOS. This includes the @ref dma,
* the @ref exceptions, the @ref interrupt and the @ref n64sys. The DMA controller
* handles DMA requests between the cartridge and the N64 RDRAM. Other systems
* in the N64 have their own DMA controllers that are handled in the relevant
* subsystems. The exception handler traps any exceptions raised by the N64,
* including the reset exception. The interrupt handler sets up the MIPS
* interface (MI) which handles low level interrupt functionality for all other
* systems in the N64. The N64 system interface provides the ability for code to
* manipulate cache and boot options.
*/
/**
* @brief Register definition for the AI interface
* @ingroup lowlevel
*/
typedef struct AI_regs_s {
/** @brief Pointer to uncached memory buffer of samples to play */
volatile void * address;
/** @brief Size in bytes of the buffer to be played. Should be
* number of stereo samples * 2 * sizeof( uint16_t )
*/
uint32_t length;
/** @brief DMA start register. Write a 1 to this register to start
* playing back an audio sample. */
uint32_t control;
/** @brief AI status register. Bit 31 is the full bit, bit 30 is the busy bit. */
uint32_t status;
/** @brief Rate at which the buffer should be played.
*
* Use the following formula to calculate the value: ((2 * clockrate / frequency) + 1) / 2 - 1
*/
uint32_t dacrate;
/** @brief The size of a single sample in bits. */
uint32_t samplesize;
} AI_regs_t;
/**
* @brief Register definition for the MI interface
* @ingroup lowlevel
*/
typedef struct MI_regs_s {
/** @brief Mode register */
uint32_t mode;
/** @brief Version register */
uint32_t version;
/** @brief Current interrupts on the system */
uint32_t intr;
/** @brief Interrupt mask */
uint32_t mask;
} MI_regs_t;
/**
* @brief Register definition for the VI interface
* @ingroup lowlevel
*/
typedef struct VI_regs_s {
/** @brief VI control register. Sets up various rasterization modes */
uint32_t control;
/** @brief Pointer to uncached buffer in memory to rasterize */
void * framebuffer;
/** @brief Width of the buffer in pixels */
uint32_t width;
/** @brief Vertical interrupt control register. Controls which horizontal
* line must be hit to generate a VI interrupt
*/
uint32_t v_int;
/** @brief Current vertical line counter. */
uint32_t cur_line;
/** @brief Timing generation register for PAL/NTSC signals */
uint32_t timing;
/** @brief Number of lines per frame */
uint32_t v_sync;
/** @brief Number of pixels in line and leap pattern */
uint32_t h_sync;
/** @brief Number of pixels in line, set identically to h_sync */
uint32_t h_sync2;
/** @brief Beginning and end of video horizontally */
uint32_t h_limits;
/** @brief Beginning and end of video vertically */
uint32_t v_limits;
/** @brief Beginning and end of color burst in vertical lines */
uint32_t color_burst;
/** @brief Horizontal scaling factor from buffer to screen */
uint32_t h_scale;
/** @brief Vertical scaling factor from buffer to screen */
uint32_t v_scale;
} VI_regs_t;
/**
* @brief Register definition for the PI interface
* @ingroup lowlevel
*/
typedef struct PI_regs_s {
/** @brief Uncached address in RAM where data should be found */
volatile void * ram_address;
/** @brief Address of data on peripheral */
uint32_t pi_address;
/** @brief How much data to read from RAM into the peripheral */
uint32_t read_length;
/** @brief How much data to write to RAM from the peripheral */
uint32_t write_length;
/** @brief Status of the PI, including DMA busy */
uint32_t status;
} PI_regs_t;
/**
* @brief Register definition for the SI interface
* @ingroup lowlevel
*/
typedef struct SI_regs_s {
/** @brief Uncached address in RAM where data should be found */
volatile void * DRAM_addr;
/** @brief Address to read when copying from PIF RAM */
volatile void * PIF_addr_read;
/** @brief Reserved word */
uint32_t reserved1;
/** @brief Reserved word */
uint32_t reserved2;
/** @brief Address to write when copying to PIF RAM */
volatile void * PIF_addr_write;
/** @brief Reserved word */
uint32_t reserved3;
/** @brief SI status, including DMA busy and IO busy */
uint32_t status;
} SI_regs_t;
/**
* @brief Register definition for the SP interface
* @ingroup lowlevel
*/
typedef struct SP_regs_s {
/** @brief RSP memory address (IMEM/DMEM) */
volatile void * RSP_addr;
/** @brief RDRAM memory address */
volatile void * RDAM_addr;
/** @brief RDRAM->RSP DMA length */
uint32_t rsp_read_length;
/** @brief RDP->RDRAM DMA length */
uint32_t rsp_write_length;
/** @brief RSP status */
uint32_t status;
/** @brief RSP DMA full */
uint32_t rsp_dma_full;
/** @brief RSP DMA busy */
uint32_t rsp_dma_busy;
/** @brief RSP Semaphore */
uint32_t rsp_semaphore;
} SP_regs_t;
#endif

View File

@ -12,7 +12,13 @@
//#include "si/cic.h"
#include <stdint.h>
#include <stddef.h>
#include <string.h>
#include <libdragon.h>
#include "regsinternal.h"
#include "cic.h"
#include "types.h"
#include "rom.h" //TODO: perhaps the pifram defines should be global
// CIC seeds and status bits passed from PIF to IPL through PIF RAM
@ -115,3 +121,94 @@ uint32_t si_crc32(const uint8_t *data, size_t size) {
return c ^ 0xFFFFFFFF;
}
static volatile struct SI_regs_s * const SI_regs = (struct SI_regs_s *) 0xa4800000;
static void * const PIF_RAM = (void *) 0x1fc007c0;
/** @brief SI DMA busy */
#define SI_STATUS_DMA_BUSY ( 1 << 0 )
/** @brief SI IO busy */
#define SI_STATUS_IO_BUSY ( 1 << 1 )
static void __SI_DMA_wait(void) {
while (SI_regs->status & (PI_STATUS_DMA_BUSY | PI_STATUS_IO_BUSY));
}
static void __controller_exec_PIF(void *inblock, void *outblock) {
volatile uint64_t inblock_temp[8];
volatile uint64_t outblock_temp[8];
data_cache_hit_writeback_invalidate(inblock_temp, 64);
memcpy(UncachedAddr(inblock_temp), inblock, 64);
/* Be sure another thread doesn't get into a resource fight */
disable_interrupts();
__SI_DMA_wait();
SI_regs->DRAM_addr = inblock_temp; // only cares about 23:0
MEMORY_BARRIER();
SI_regs->PIF_addr_write = PIF_RAM; // is it really ever anything else?
MEMORY_BARRIER();
__SI_DMA_wait();
data_cache_hit_writeback_invalidate(outblock_temp, 64);
SI_regs->DRAM_addr = outblock_temp;
MEMORY_BARRIER();
SI_regs->PIF_addr_read = PIF_RAM;
MEMORY_BARRIER();
__SI_DMA_wait();
/* Now that we've copied, its safe to let other threads go */
enable_interrupts();
memcpy(outblock, UncachedAddr(outblock_temp), 64);
}
int pifram_x105_response_test() {
static unsigned long long SI_eeprom_read_block[8] = {
0xFFFFFFFFFFFFFFFF,
0xFFFFFFFFFFFFFFFF,
0xFFFFFFFFFFFFFFFF,
0xFFFFFFFFFFFFFFFF,
0xFFFFFFFFFFFFFFFF,
0xFFFFFFFFFFFF0F0F,
0x8B00620018000600,
0x0100C000B0000002 //0x3f=02
};
static unsigned long long output[8];
__controller_exec_PIF(SI_eeprom_read_block,output);
/*
expected result
FF FF FF FF FF FF FF FF
FF FF FF FF FF FF FF FF
FF FF FF FF FF FF FF FF
FF FF FF FF FF FF FF FF
FF FF FF FF FF FF FF FF
FF FF FF FF FF FF 00 00
3E C6 C0 4E BD 37 15 55
5A 8C 2A 8C D3 71 71 00
*/
/* We are looking for 0x55 in [6], which
* signifies that there is an x105 present.*/
if( (output[6] & 0xFF) == 0x55 )
{
//x105 found!
return 1;
} else {
//x105 not found!
return 0;
}
}