C语言实现ARM MCUJTAG离线调试器

以下是一个使用C语言实现ARM MCU JTAG离线调试器的示例代码,该代码可以读取目标设备的寄存器和内存:

复制代码
#include <stdio.h>
#include <stdlib.h>
#include <stdint.h>
#include <string.h>

#define TCK_PIN 0
#define TMS_PIN 1
#define TDI_PIN 2
#define TDO_PIN 3

#define TCK_H() GPIO_SetBits(GPIOA, GPIO_Pin_0)
#define TCK_L() GPIO_ResetBits(GPIOA, GPIO_Pin_0)
#define TMS_H() GPIO_SetBits(GPIOA, GPIO_Pin_1)
#define TMS_L() GPIO_ResetBits(GPIOA, GPIO_Pin_1)
#define TDI_H() GPIO_SetBits(GPIOA, GPIO_Pin_2)
#define TDI_L() GPIO_ResetBits(GPIOA, GPIO_Pin_2)

#define TDO_IN() GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING
#define TDO_OUT() GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP

GPIO_InitTypeDef GPIO_InitStructure;

void JTAG_Init(void);
void JTAG_SendData(uint8_t *data, uint32_t len);
void JTAG_ReceiveData(uint8_t *data, uint32_t len);
void JTAG_ReadReg(uint32_t addr, uint32_t *data);
void JTAG_WriteReg(uint32_t addr, uint32_t data);
void JTAG_ReadMem(uint32_t addr, uint8_t *data, uint32_t len);
void JTAG_WriteMem(uint32_t addr, uint8_t *data, uint32_t len);

int main(void)
{
    uint32_t reg_value;
    uint8_t mem_data[256];

    // 初始化GPIO
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_3;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    // 初始化JTAG接口
    JTAG_Init();

    // 读取寄存器
    JTAG_ReadReg(0xE000EDFC, &reg_value);
    printf("SP: 0x%08X\n", reg_value);
    JTAG_ReadReg(0xE000EDF0, &reg_value);
    printf("PC: 0x%08X\n", reg_value);

    // 读取内存
    JTAG_ReadMem(0x08000000, mem_data, sizeof(mem_data));
    for (int i = 0; i < sizeof(mem_data); i += 4) {
        printf("0x%08X: %02X %02X %02X %02X\n",
            0x08000000 + i,
            mem_data[i], mem_data[i+1], mem_data[i+2], mem_data[i+3]);
    }

    return 0;
}

void JTAG_Init(void)
{
    // 发送复位序列
    TCK_L();
    TMS_H();
    TDI_L();
    for (int i = 0; i < 5; i++) {
        TCK_H();
        TCK_L();
    }

    // 发送Test-Logic-Reset序列
    TCK_L();
    TMS_H();
    TCK_H();
    TCK_L();
    TCK_H();
    TCK_L();
    TCK_H();
    TCK_L();
    TMS_L();
    TCK_H();
    TCK_L();

    // 发送Run-Test/Idle序列
    for (int i = 0; i < 10; i++) {
        TCK_H();
        TCK_L();
    }
}

void JTAG_SendData(uint8_t *data, uint32_t len)
{
    for (int i = 0; i < len; i++) {
        for (int j = 0; j < 8; j++) {
            if (data[i] & (1 << j)) {
                TDI_H();
            } else {
                TDI_L();
            }

            TCK_H();
            TCK_L();
        }
    }
}

void JTAG_ReceiveData(uint8_t *data, uint32_t len)
{
    for (int i = 0; i < len; i++) {
        data[i] = 0;

        for (int j = 0; j < 8; j++) {
            TCK_H();
            data[i] |= GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_3) << j;
            TCK_L();
        }
    }
}

void JTAG_ReadReg(uint32_t addr, uint32_t *data)
{
    uint8_t req[5];
    uint8_t res[4];

    req[0] = 0x8E;
    req[1] = (addr >> 16) & 0xFF;
    req[2] = (addr >> 8) & 0xFF;
    req[3] = (addr >> 0) & 0xFF;
    req[4] = 0x00;

    JTAG_SendData(req, sizeof(req));
    JTAG_ReceiveData(res, sizeof(res));

    *data = (res[0] << 0) | (res[1] << 8) | (res[2] << 16) | (res[3] << 24);
}

void JTAG_WriteReg(uint32_t addr, uint32_t data)
{
    uint8_t req[9];

    req[0] = 0xAE;
    req[1] = (addr >> 16) & 0xFF;
    req[2] = (addr >> 8) & 0xFF;
    req[3] = (addr >> 0) & 0xFF;
    req[4] = (data >> 24) & 0xFF;
    req[5] = (data >> 16) & 0xFF;
    req[6] = (data >> 8) & 0xFF;
    req[7] = (data >> 0) & 0xFF;
    req[8] = 0x00;

    JTAG_SendData(req, sizeof(req));
}

void JTAG_ReadMem(uint32_t addr, uint8_t *data, uint32_t len)
{
    uint8_t req[7];

    req[0] = 0x9E;
    req[1] = (addr >> 16) & 0xFF;
    req[2] = (addr >> 8) & 0xFF;
    req[3] = (addr >> 0) & 0xFF;
    req[4] = (len >> 16) & 0xFF;
    req[5] = (len >> 8) & 0xFF;
    req[6] = (len >> 0) & 0xFF;

    JTAG_SendData(req, sizeof(req));
    JTAG_ReceiveData(data, len);
}

void JTAG_WriteMem(uint32_t addr, uint8_t *data, uint32_t len)
{
    uint8_t req[7];

    req[0] = 0xBE;
    req[1] = (addr >> 16) & 0xFF;
    req[2] = (addr >> 8) & 0xFF;
    req[3] = (addr >> 0) & 0xFF;
    req[4] = (len >> 16) & 0xFF;
    req[5] = (len >> 8) & 0xFF;
    req[6] = (len >> 0) & 0xFF;

    JTAG_SendData(req, sizeof(req));
    JTAG_SendData(data, len);
}

该代码使用STM32F103C8T6作为调试器,通过JTAG接口连接目标设备。调试器通过GPIO控制JTAG接口的时钟和数据线,实现JTAG协议的相关功能。该代码实现了读取寄存器和读取内存的功能,可以用于基本的ARM MCU离线调试。需要注意的是,该代码仅供参考,实际应用需要根据具体的需求进行修改和优化。

相关推荐
小美单片机5 分钟前
External model DLL ”ADC083XDLL“ not found_proteus仿真报错解决方法
c语言·单片机·51单片机·proteus·课程设计·课设
Tyrion.Mon17 分钟前
沁恒PD协议诱骗芯片CH224A/CH224Q--IIC研究(0x60~0x8F)
单片机·硬件工程
程芯带你刷C语言简单算法题19 分钟前
Day48~对于高度为 n 的台阶,从下往上走,每一步的阶数为 1,2,3 中的一个。问要走到顶部一共有多少种走法
c语言·开发语言·学习·算法·c
田甲28 分钟前
基于STM32L051和HDC2080的低功耗温湿度计
单片机·嵌入式硬件·温湿度计·hdc2080
List<String> error_P1 小时前
STM32 GPIO HAL库常用函数
stm32·单片机·hal库
皮蛋sol周1 小时前
嵌入式学习数据结构(三)栈 链式 循环队列
arm开发·数据结构·学习·算法··循环队列·链式队列
进击的小头1 小时前
为什么C语言也需要设计模式
c语言·开发语言·设计模式
啟明起鸣2 小时前
【Linux 项目管理工具】GDB 调试是现成 C/C++ 项目的 “造影剂”,用来分析项目的架构原理
linux·c语言·c++
小痞同学2 小时前
【铁头山羊STM32】HAL库 5.SPI部分
stm32·单片机·嵌入式硬件
蓬荜生灰2 小时前
STM32(5)-- 新建寄存器版工程
stm32·单片机·嵌入式硬件