rockbox/firmware/target/arm/stm32/debug-stm32h7.c
Aidan MacDonald 84fa538979 echoplayer: add PCM debug menu
Display DMA error counters and the SAI underrun counter
in the debug menu.

Change-Id: I235bfcb0fa965d87c30deeb300535141eda5f974
2026-03-03 09:24:01 -05:00

193 lines
5 KiB
C

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2025 by Aidan MacDonald
*
* This program 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 2
* of the License, or (at your option) any later version.
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
* KIND, either express or implied.
*
****************************************************************************/
#include "system.h"
#include "kernel.h"
#include "button.h"
#include "lcd.h"
#include "font.h"
#include "action.h"
#include "list.h"
#include "regs/stm32h743/dbgmcu.h"
#include "regs/cortex-m/cm_debug.h"
#include <stdio.h>
enum
{
SWD_MENU_IS_ENABLED,
SWD_MENU_IS_ATTACHED,
SWD_MENU_NUM_ENTRIES
};
static int swd_menu_action_cb(int action, struct gui_synclist *lists)
{
int sel = gui_synclist_get_sel_pos(lists);
if (sel == SWD_MENU_IS_ENABLED && action == ACTION_STD_OK)
{
if (reg_readf(DBGMCU_CR, DBGSLEEP_D1))
system_debug_enable(false);
else
system_debug_enable(true);
action = ACTION_REDRAW;
}
if (action == ACTION_NONE)
action = ACTION_REDRAW;
return action;
}
static const char *swd_menu_get_name(int item, void *data, char *buf, size_t bufsz)
{
(void)data;
switch (item)
{
case SWD_MENU_IS_ENABLED:
snprintf(buf, bufsz, "System debug enabled: %d",
(int)!!reg_readf(DBGMCU_CR, DBGSLEEP_D1));
return buf;
case SWD_MENU_IS_ATTACHED:
snprintf(buf, bufsz, "DHCSR.C_DEBUGEN bit: %d",
(int)!!reg_readf(CM_DEBUG_DHCSR, C_DEBUGEN));
return buf;
default:
return "";
}
}
static bool swd_menu(void)
{
struct simplelist_info info;
simplelist_info_init(&info, "SWD/JTAG", SWD_MENU_NUM_ENTRIES, NULL);
info.action_callback = swd_menu_action_cb;
info.get_name = swd_menu_get_name;
return simplelist_show_list(&info);
}
#if defined(ECHO_R1)
extern volatile int pcm_sai_xrun_count;
extern volatile int pcm_dma_teif_count;
extern volatile int pcm_dma_dmeif_count;
extern volatile int pcm_dma_feif_count;
struct pcmdebug_counter
{
const char *name;
volatile int *value;
};
static const struct pcmdebug_counter pcmdebug_counters[] =
{
{"SAI xrun count", &pcm_sai_xrun_count},
{"DMA TEIF count", &pcm_dma_teif_count},
{"DMA DMEIF count", &pcm_dma_dmeif_count},
{"DMA FEIF count", &pcm_dma_feif_count},
};
static int pcmdebug_menu_action_cb(int action, struct gui_synclist *lists)
{
if (action == ACTION_STD_OK)
{
int item = gui_synclist_get_sel_pos(lists);
const struct pcmdebug_counter *counter = &pcmdebug_counters[item];
*counter->value = 0;
action = ACTION_REDRAW;
}
if (action == ACTION_NONE)
action = ACTION_REDRAW;
return action;
}
static const char *pcmdebug_menu_get_name(int item, void *data, char *buf, size_t bufsz)
{
const struct pcmdebug_counter *counter = &pcmdebug_counters[item];
(void)data;
snprintf(buf, bufsz, "%s: %d", counter->name, *counter->value);
return buf;
}
static bool pcm_debug_menu(void)
{
struct simplelist_info info;
simplelist_info_init(&info, "PCM debug", ARRAYLEN(pcmdebug_counters), NULL);
info.action_callback = pcmdebug_menu_action_cb;
info.get_name = pcmdebug_menu_get_name;
return simplelist_show_list(&info);
}
#endif
/* Menu definition */
static const struct {
const char *name;
bool (*function) (void);
} menuitems[] = {
{"PCM debug", pcm_debug_menu},
{"SWD/JTAG", swd_menu},
};
static int hw_info_menu_action_cb(int btn, struct gui_synclist *lists)
{
if (btn == ACTION_STD_OK)
{
int sel = gui_synclist_get_sel_pos(lists);
FOR_NB_SCREENS(i)
viewportmanager_theme_enable(i, false, NULL);
menuitems[sel].function();
btn = ACTION_REDRAW;
FOR_NB_SCREENS(i)
viewportmanager_theme_undo(i, false);
}
return btn;
}
static const char* hw_info_menu_get_name(int item, void *data, char *buf, size_t bufsz)
{
(void)buf;
(void)bufsz;
(void)data;
return menuitems[item].name;
}
bool dbg_hw_info(void)
{
struct simplelist_info info;
simplelist_info_init(&info, MODEL_NAME " debug menu",
ARRAYLEN(menuitems), NULL);
info.action_callback = hw_info_menu_action_cb;
info.get_name = hw_info_menu_get_name;
return simplelist_show_list(&info);
}
bool dbg_ports(void)
{
return false;
}