Compare commits
4 Commits
89279e27d5
...
dev
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
31aeeaabbb | ||
| 89994995af | |||
| 15b7cb1f4c | |||
| 4b124a9998 |
2
.github/workflows/build.yml
vendored
2
.github/workflows/build.yml
vendored
@@ -1,7 +1,7 @@
|
|||||||
name: Build
|
name: Build
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
#push:
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
|||||||
@@ -164,6 +164,12 @@ void bq24193_enable_charger()
|
|||||||
i2c_send_byte(I2C_1, BQ24193_I2C_ADDR, BQ24193_PORConfig, reg);
|
i2c_send_byte(I2C_1, BQ24193_I2C_ADDR, BQ24193_PORConfig, reg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool bq24193_charger_connected()
|
||||||
|
{
|
||||||
|
u8 status = bq24193_get_reg(BQ24193_Status);
|
||||||
|
return (status & BQ24193_STATUS_PG_MASK) != 0;
|
||||||
|
}
|
||||||
|
|
||||||
void bq24193_fake_battery_removal()
|
void bq24193_fake_battery_removal()
|
||||||
{
|
{
|
||||||
// Disable watchdog to keep BATFET disabled.
|
// Disable watchdog to keep BATFET disabled.
|
||||||
|
|||||||
@@ -19,6 +19,8 @@
|
|||||||
#ifndef __BQ24193_H_
|
#ifndef __BQ24193_H_
|
||||||
#define __BQ24193_H_
|
#define __BQ24193_H_
|
||||||
|
|
||||||
|
#include <utils/types.h>
|
||||||
|
|
||||||
#define BQ24193_I2C_ADDR 0x6B
|
#define BQ24193_I2C_ADDR 0x6B
|
||||||
|
|
||||||
// REG 0 masks.
|
// REG 0 masks.
|
||||||
@@ -117,5 +119,6 @@ enum BQ24193_reg_prop {
|
|||||||
int bq24193_get_property(enum BQ24193_reg_prop prop, int *value);
|
int bq24193_get_property(enum BQ24193_reg_prop prop, int *value);
|
||||||
void bq24193_enable_charger();
|
void bq24193_enable_charger();
|
||||||
void bq24193_fake_battery_removal();
|
void bq24193_fake_battery_removal();
|
||||||
|
bool bq24193_charger_connected();
|
||||||
|
|
||||||
#endif /* __BQ24193_H_ */
|
#endif /* __BQ24193_H_ */
|
||||||
|
|||||||
@@ -6,14 +6,97 @@
|
|||||||
#include "fs.h"
|
#include "fs.h"
|
||||||
#include <libs/fatfs/ff.h>
|
#include <libs/fatfs/ff.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <stdarg.h>
|
||||||
#include <utils/sprintf.h>
|
#include <utils/sprintf.h>
|
||||||
|
|
||||||
|
#define BACKUP_PATH_MAX 270
|
||||||
|
|
||||||
// Check if file/directory exists
|
// Check if file/directory exists
|
||||||
static bool path_exists(const char *path) {
|
static bool path_exists(const char *path) {
|
||||||
FILINFO fno;
|
FILINFO fno;
|
||||||
return (f_stat(path, &fno) == FR_OK);
|
return (f_stat(path, &fno) == FR_OK);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Create all parent directories for a path (mkdir -p style)
|
||||||
|
static void ensure_parent_dirs(const char *full_path) {
|
||||||
|
char buf[BACKUP_PATH_MAX];
|
||||||
|
int n = (int)strlen(full_path);
|
||||||
|
if (n >= BACKUP_PATH_MAX) return;
|
||||||
|
memcpy(buf, full_path, n + 1);
|
||||||
|
|
||||||
|
for (int i = 4; i < n; i++) {
|
||||||
|
if (buf[i] == '/') {
|
||||||
|
buf[i] = '\0';
|
||||||
|
f_mkdir(buf);
|
||||||
|
buf[i] = '/';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Copy a single path (file or dir) to backup base, preserving SD structure
|
||||||
|
static int backup_single_path(const char *src_path, const char *backup_base) {
|
||||||
|
FILINFO fno;
|
||||||
|
if (f_stat(src_path, &fno) != FR_OK)
|
||||||
|
return FR_OK;
|
||||||
|
|
||||||
|
const char *relative = src_path;
|
||||||
|
if (strncmp(relative, "sd:/", 4) == 0)
|
||||||
|
relative += 4;
|
||||||
|
|
||||||
|
if (fno.fattrib & AM_DIR) {
|
||||||
|
const char *last_slash = strrchr(relative, '/');
|
||||||
|
char parent[BACKUP_PATH_MAX];
|
||||||
|
if (last_slash && last_slash > relative) {
|
||||||
|
s_printf(parent, "%s/%.*s", backup_base, (int)(last_slash - relative), relative);
|
||||||
|
/* Create full parent hierarchy (e.g. atmosphere, atmosphere/contents) */
|
||||||
|
ensure_parent_dirs(parent);
|
||||||
|
f_mkdir(parent); /* Ensure parent exists; FR_EXIST is fine */
|
||||||
|
return folder_copy(src_path, parent);
|
||||||
|
} else {
|
||||||
|
ensure_parent_dirs(backup_base);
|
||||||
|
return folder_copy(src_path, backup_base);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
char dst_path[BACKUP_PATH_MAX];
|
||||||
|
s_printf(dst_path, "%s/%s", backup_base, relative);
|
||||||
|
ensure_parent_dirs(dst_path);
|
||||||
|
return file_copy(src_path, dst_path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Backup all paths from varargs lists (same structure as delete_path_lists_grouped)
|
||||||
|
int backup_deletion_lists(const char *backup_base, ...) {
|
||||||
|
va_list ap;
|
||||||
|
const char **list;
|
||||||
|
int res = FR_OK;
|
||||||
|
|
||||||
|
/* FatFS f_mkdir only creates the last component; parent must exist first */
|
||||||
|
ensure_parent_dirs(BACKUP_BASE_PATH);
|
||||||
|
res = f_mkdir(BACKUP_BASE_PATH);
|
||||||
|
if (res != FR_OK && res != FR_EXIST)
|
||||||
|
return res;
|
||||||
|
|
||||||
|
ensure_parent_dirs(backup_base);
|
||||||
|
res = f_mkdir(backup_base);
|
||||||
|
if (res != FR_OK && res != FR_EXIST)
|
||||||
|
return res;
|
||||||
|
|
||||||
|
va_start(ap, backup_base);
|
||||||
|
while ((list = va_arg(ap, const char **)) != NULL) {
|
||||||
|
for (int i = 0; list[i] != NULL; i++) {
|
||||||
|
if (path_exists(list[i])) {
|
||||||
|
int copy_res = backup_single_path(list[i], backup_base);
|
||||||
|
if (copy_res != FR_OK && res == FR_OK)
|
||||||
|
res = copy_res; // Remember first error but keep going
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
va_end(ap);
|
||||||
|
|
||||||
|
/* Best-effort: return FR_OK so install continues even if some copies failed */
|
||||||
|
return FR_OK;
|
||||||
|
}
|
||||||
|
|
||||||
// Backup user data before clean install
|
// Backup user data before clean install
|
||||||
int backup_user_data(void) {
|
int backup_user_data(void) {
|
||||||
int res;
|
int res;
|
||||||
|
|||||||
@@ -7,6 +7,8 @@
|
|||||||
#include <utils/types.h>
|
#include <utils/types.h>
|
||||||
|
|
||||||
#define TEMP_BACKUP_PATH "sd:/temp_backup"
|
#define TEMP_BACKUP_PATH "sd:/temp_backup"
|
||||||
|
#define BACKUP_BASE_PATH "sd:/backup/OmniNX"
|
||||||
|
#define PRE_OMNINX_LABEL "pre-omninx"
|
||||||
|
|
||||||
// Backup user data (DBI, Tinfoil, prod.keys) before clean install
|
// Backup user data (DBI, Tinfoil, prod.keys) before clean install
|
||||||
int backup_user_data(void);
|
int backup_user_data(void);
|
||||||
@@ -16,3 +18,6 @@ int restore_user_data(void);
|
|||||||
|
|
||||||
// Clean up temporary backup directory
|
// Clean up temporary backup directory
|
||||||
int cleanup_backup(void);
|
int cleanup_backup(void);
|
||||||
|
|
||||||
|
// Backup paths from varargs lists to backup_base (pass path arrays, terminate with NULL)
|
||||||
|
int backup_deletion_lists(const char *backup_base, ...);
|
||||||
|
|||||||
@@ -591,9 +591,29 @@ int perform_installation(omninx_variant_t pack_variant, install_mode_t mode) {
|
|||||||
int res;
|
int res;
|
||||||
|
|
||||||
if (mode == INSTALL_MODE_UPDATE) {
|
if (mode == INSTALL_MODE_UPDATE) {
|
||||||
// Update mode: selective cleanup then install
|
// Update mode: backup, selective cleanup, then install
|
||||||
|
omninx_status_t status = detect_omninx_installation();
|
||||||
|
const char *version = (status.version_file[0] != '\0') ? status.version_file : "unknown";
|
||||||
|
|
||||||
install_set_color(COLOR_YELLOW);
|
install_set_color(COLOR_YELLOW);
|
||||||
gfx_printf("Schritt 1: Bereinigung...\n");
|
gfx_printf("Schritt 1: Backup vor Update (%s)...\n", version);
|
||||||
|
install_set_color(COLOR_WHITE);
|
||||||
|
res = backup_before_update(version);
|
||||||
|
/* Backup is best-effort; continue install even on failure */
|
||||||
|
if (res == FR_OK) {
|
||||||
|
install_set_color(COLOR_GREEN);
|
||||||
|
gfx_printf(" [OK] Sicherung nach sd:/backup/OmniNX/%s/\n", version);
|
||||||
|
install_set_color(COLOR_WHITE);
|
||||||
|
} else {
|
||||||
|
install_set_color(COLOR_ORANGE);
|
||||||
|
gfx_printf(" [WARN] Backup fehlgeschlagen, fahre trotzdem fort...\n");
|
||||||
|
install_set_color(COLOR_WHITE);
|
||||||
|
}
|
||||||
|
|
||||||
|
install_check_and_clear_screen_if_needed();
|
||||||
|
gfx_printf("\n");
|
||||||
|
install_set_color(COLOR_YELLOW);
|
||||||
|
gfx_printf("Schritt 2: Bereinigung...\n");
|
||||||
install_set_color(COLOR_WHITE);
|
install_set_color(COLOR_WHITE);
|
||||||
res = update_mode_cleanup(pack_variant);
|
res = update_mode_cleanup(pack_variant);
|
||||||
if (res != FR_OK) return res;
|
if (res != FR_OK) return res;
|
||||||
@@ -601,7 +621,7 @@ int perform_installation(omninx_variant_t pack_variant, install_mode_t mode) {
|
|||||||
install_check_and_clear_screen_if_needed();
|
install_check_and_clear_screen_if_needed();
|
||||||
gfx_printf("\n");
|
gfx_printf("\n");
|
||||||
install_set_color(COLOR_YELLOW);
|
install_set_color(COLOR_YELLOW);
|
||||||
gfx_printf("Schritt 2: Dateien kopieren...\n");
|
gfx_printf("Schritt 3: Dateien kopieren...\n");
|
||||||
install_set_color(COLOR_WHITE);
|
install_set_color(COLOR_WHITE);
|
||||||
res = update_mode_install(pack_variant);
|
res = update_mode_install(pack_variant);
|
||||||
if (res != FR_OK) return res;
|
if (res != FR_OK) return res;
|
||||||
@@ -614,7 +634,7 @@ int perform_installation(omninx_variant_t pack_variant, install_mode_t mode) {
|
|||||||
res = cleanup_other_staging_directories(pack_variant);
|
res = cleanup_other_staging_directories(pack_variant);
|
||||||
return res;
|
return res;
|
||||||
} else {
|
} else {
|
||||||
// Clean mode: backup, wipe, restore, install
|
// Clean mode: backup user data, backup pre-omninx, wipe, restore, install
|
||||||
install_set_color(COLOR_YELLOW);
|
install_set_color(COLOR_YELLOW);
|
||||||
gfx_printf("Schritt 1: Sichere Benutzerdaten...\n");
|
gfx_printf("Schritt 1: Sichere Benutzerdaten...\n");
|
||||||
install_set_color(COLOR_WHITE);
|
install_set_color(COLOR_WHITE);
|
||||||
@@ -624,7 +644,23 @@ int perform_installation(omninx_variant_t pack_variant, install_mode_t mode) {
|
|||||||
install_check_and_clear_screen_if_needed();
|
install_check_and_clear_screen_if_needed();
|
||||||
gfx_printf("\n");
|
gfx_printf("\n");
|
||||||
install_set_color(COLOR_YELLOW);
|
install_set_color(COLOR_YELLOW);
|
||||||
gfx_printf("Schritt 2: Bereinige alte Installation...\n");
|
gfx_printf("Schritt 2: Backup alter CFW-Dateien (pre-omninx)...\n");
|
||||||
|
install_set_color(COLOR_WHITE);
|
||||||
|
res = backup_before_clean();
|
||||||
|
if (res == FR_OK) {
|
||||||
|
install_set_color(COLOR_GREEN);
|
||||||
|
gfx_printf(" [OK] Sicherung nach sd:/backup/OmniNX/pre-omninx/\n");
|
||||||
|
install_set_color(COLOR_WHITE);
|
||||||
|
} else {
|
||||||
|
install_set_color(COLOR_ORANGE);
|
||||||
|
gfx_printf(" [WARN] Backup fehlgeschlagen, fahre trotzdem fort...\n");
|
||||||
|
install_set_color(COLOR_WHITE);
|
||||||
|
}
|
||||||
|
|
||||||
|
install_check_and_clear_screen_if_needed();
|
||||||
|
gfx_printf("\n");
|
||||||
|
install_set_color(COLOR_YELLOW);
|
||||||
|
gfx_printf("Schritt 3: Bereinige alte Installation...\n");
|
||||||
install_set_color(COLOR_WHITE);
|
install_set_color(COLOR_WHITE);
|
||||||
res = clean_mode_wipe();
|
res = clean_mode_wipe();
|
||||||
if (res != FR_OK) return res;
|
if (res != FR_OK) return res;
|
||||||
@@ -632,7 +668,7 @@ int perform_installation(omninx_variant_t pack_variant, install_mode_t mode) {
|
|||||||
install_check_and_clear_screen_if_needed();
|
install_check_and_clear_screen_if_needed();
|
||||||
gfx_printf("\n");
|
gfx_printf("\n");
|
||||||
install_set_color(COLOR_YELLOW);
|
install_set_color(COLOR_YELLOW);
|
||||||
gfx_printf("Schritt 3: Stelle Benutzerdaten wieder her...\n");
|
gfx_printf("Schritt 4: Stelle Benutzerdaten wieder her...\n");
|
||||||
install_set_color(COLOR_WHITE);
|
install_set_color(COLOR_WHITE);
|
||||||
res = clean_mode_restore();
|
res = clean_mode_restore();
|
||||||
if (res != FR_OK) return res;
|
if (res != FR_OK) return res;
|
||||||
@@ -640,7 +676,7 @@ int perform_installation(omninx_variant_t pack_variant, install_mode_t mode) {
|
|||||||
install_check_and_clear_screen_if_needed();
|
install_check_and_clear_screen_if_needed();
|
||||||
gfx_printf("\n");
|
gfx_printf("\n");
|
||||||
install_set_color(COLOR_YELLOW);
|
install_set_color(COLOR_YELLOW);
|
||||||
gfx_printf("Schritt 4: Dateien kopieren...\n");
|
gfx_printf("Schritt 5: Dateien kopieren...\n");
|
||||||
install_set_color(COLOR_WHITE);
|
install_set_color(COLOR_WHITE);
|
||||||
res = clean_mode_install(pack_variant);
|
res = clean_mode_install(pack_variant);
|
||||||
if (res != FR_OK) return res;
|
if (res != FR_OK) return res;
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ typedef enum {
|
|||||||
int perform_installation(omninx_variant_t pack_variant, install_mode_t mode);
|
int perform_installation(omninx_variant_t pack_variant, install_mode_t mode);
|
||||||
|
|
||||||
// Update mode operations (install_update.c)
|
// Update mode operations (install_update.c)
|
||||||
|
int backup_before_update(const char *version);
|
||||||
int update_mode_cleanup(omninx_variant_t variant);
|
int update_mode_cleanup(omninx_variant_t variant);
|
||||||
int update_mode_install(omninx_variant_t variant);
|
int update_mode_install(omninx_variant_t variant);
|
||||||
int cleanup_staging_directory(omninx_variant_t pack_variant);
|
int cleanup_staging_directory(omninx_variant_t pack_variant);
|
||||||
@@ -24,6 +25,7 @@ int cleanup_staging_directory(omninx_variant_t pack_variant);
|
|||||||
int cleanup_other_staging_directories(omninx_variant_t installed_variant);
|
int cleanup_other_staging_directories(omninx_variant_t installed_variant);
|
||||||
|
|
||||||
// Clean install operations (install_clean.c)
|
// Clean install operations (install_clean.c)
|
||||||
|
int backup_before_clean(void);
|
||||||
int clean_mode_backup(void);
|
int clean_mode_backup(void);
|
||||||
int clean_mode_wipe(void);
|
int clean_mode_wipe(void);
|
||||||
int clean_mode_restore(void);
|
int clean_mode_restore(void);
|
||||||
|
|||||||
@@ -11,6 +11,7 @@
|
|||||||
#include "gfx.h"
|
#include "gfx.h"
|
||||||
#include "version.h"
|
#include "version.h"
|
||||||
#include <libs/fatfs/ff.h>
|
#include <libs/fatfs/ff.h>
|
||||||
|
#include <utils/sprintf.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
#undef COLOR_CYAN
|
#undef COLOR_CYAN
|
||||||
@@ -31,6 +32,26 @@
|
|||||||
#define path_exists install_path_exists
|
#define path_exists install_path_exists
|
||||||
#define count_directory_items install_count_directory_items
|
#define count_directory_items install_count_directory_items
|
||||||
|
|
||||||
|
// Backup paths before clean install (sd:/backup/OmniNX/pre-omninx)
|
||||||
|
int backup_before_clean(void) {
|
||||||
|
char backup_base[270];
|
||||||
|
s_printf(backup_base, "%s/%s", BACKUP_BASE_PATH, PRE_OMNINX_LABEL);
|
||||||
|
return backup_deletion_lists(backup_base,
|
||||||
|
clean_atmosphere_dirs_to_delete,
|
||||||
|
clean_atmosphere_contents_dirs_to_delete,
|
||||||
|
clean_atmosphere_files_to_delete,
|
||||||
|
clean_bootloader_dirs_to_delete,
|
||||||
|
clean_bootloader_files_to_delete,
|
||||||
|
clean_config_dirs_to_delete,
|
||||||
|
clean_switch_dirs_to_delete,
|
||||||
|
clean_switch_files_to_delete,
|
||||||
|
clean_root_files_to_delete,
|
||||||
|
clean_misc_dirs_to_delete,
|
||||||
|
clean_misc_files_to_delete,
|
||||||
|
old_version_files_to_delete,
|
||||||
|
NULL);
|
||||||
|
}
|
||||||
|
|
||||||
// Clean mode: Backup user data
|
// Clean mode: Backup user data
|
||||||
int clean_mode_backup(void) {
|
int clean_mode_backup(void) {
|
||||||
set_color(COLOR_CYAN);
|
set_color(COLOR_CYAN);
|
||||||
|
|||||||
@@ -4,6 +4,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "install.h"
|
#include "install.h"
|
||||||
|
#include "backup.h"
|
||||||
#include "deletion_lists_update.h"
|
#include "deletion_lists_update.h"
|
||||||
#include "fs.h"
|
#include "fs.h"
|
||||||
#include "version.h"
|
#include "version.h"
|
||||||
@@ -34,6 +35,25 @@
|
|||||||
#define path_exists install_path_exists
|
#define path_exists install_path_exists
|
||||||
#define count_directory_items install_count_directory_items
|
#define count_directory_items install_count_directory_items
|
||||||
|
|
||||||
|
// Backup paths before update (sd:/backup/OmniNX/{version})
|
||||||
|
int backup_before_update(const char *version) {
|
||||||
|
char backup_base[270];
|
||||||
|
s_printf(backup_base, "%s/%s", BACKUP_BASE_PATH, version);
|
||||||
|
return backup_deletion_lists(backup_base,
|
||||||
|
atmosphere_dirs_to_delete,
|
||||||
|
atmosphere_contents_dirs_to_delete,
|
||||||
|
atmosphere_files_to_delete,
|
||||||
|
bootloader_dirs_to_delete,
|
||||||
|
bootloader_files_to_delete,
|
||||||
|
config_dirs_to_delete,
|
||||||
|
switch_dirs_to_delete,
|
||||||
|
switch_files_to_delete,
|
||||||
|
root_files_to_delete,
|
||||||
|
misc_dirs_to_delete,
|
||||||
|
misc_files_to_delete,
|
||||||
|
NULL);
|
||||||
|
}
|
||||||
|
|
||||||
// Update mode: Cleanup specific directories/files
|
// Update mode: Cleanup specific directories/files
|
||||||
int update_mode_cleanup(omninx_variant_t variant) {
|
int update_mode_cleanup(omninx_variant_t variant) {
|
||||||
(void)variant;
|
(void)variant;
|
||||||
|
|||||||
@@ -18,6 +18,8 @@
|
|||||||
#include <mem/heap.h>
|
#include <mem/heap.h>
|
||||||
#include <mem/minerva.h>
|
#include <mem/minerva.h>
|
||||||
#include <power/max77620.h>
|
#include <power/max77620.h>
|
||||||
|
#include <power/max17050.h>
|
||||||
|
#include <power/bq24193.h>
|
||||||
#include <soc/bpmp.h>
|
#include <soc/bpmp.h>
|
||||||
#include <soc/fuse.h>
|
#include <soc/fuse.h>
|
||||||
#include <soc/hw_init.h>
|
#include <soc/hw_init.h>
|
||||||
@@ -376,6 +378,56 @@ void ipl_main(void) {
|
|||||||
// Determine installation mode
|
// Determine installation mode
|
||||||
install_mode_t mode = current.is_installed ? INSTALL_MODE_UPDATE : INSTALL_MODE_CLEAN;
|
install_mode_t mode = current.is_installed ? INSTALL_MODE_UPDATE : INSTALL_MODE_CLEAN;
|
||||||
|
|
||||||
|
// Battery run protection: below 10% requires charger
|
||||||
|
#define BATT_LOW_THRESHOLD 10
|
||||||
|
int batt_raw = 0;
|
||||||
|
int batt_pct = 100;
|
||||||
|
if (max17050_get_property(MAX17050_RepSOC, &batt_raw) == 0) {
|
||||||
|
batt_pct = batt_raw >> 8; // RepSOC: high byte = percent
|
||||||
|
}
|
||||||
|
while (batt_pct < BATT_LOW_THRESHOLD && !bq24193_charger_connected()) {
|
||||||
|
gfx_clear_grey(0x1B);
|
||||||
|
gfx_con_setpos(0, 0);
|
||||||
|
print_header();
|
||||||
|
set_color(COLOR_RED);
|
||||||
|
gfx_printf("Akkustand zu niedrig! (%d%%)\n\n", batt_pct);
|
||||||
|
gfx_printf("Unter %d%% Akku darf die Installation nur\n", BATT_LOW_THRESHOLD);
|
||||||
|
gfx_printf("mit angeschlossenem Ladegeraet durchgefuehrt werden.\n\n");
|
||||||
|
set_color(COLOR_YELLOW);
|
||||||
|
gfx_printf("Stecke das Ladegeraet an und warte...\n");
|
||||||
|
gfx_printf("Oder druecke + und - zum Abbrechen.\n");
|
||||||
|
set_color(COLOR_WHITE);
|
||||||
|
jc_init_hw();
|
||||||
|
while (btn_read() & BTN_POWER) { msleep(50); }
|
||||||
|
bool user_cancelled = false;
|
||||||
|
while (batt_pct < BATT_LOW_THRESHOLD && !bq24193_charger_connected() && !user_cancelled) {
|
||||||
|
jc_gamepad_rpt_t *jc = joycon_poll();
|
||||||
|
if (jc && jc->plus && jc->minus) {
|
||||||
|
user_cancelled = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (max17050_get_property(MAX17050_RepSOC, &batt_raw) == 0) {
|
||||||
|
batt_pct = batt_raw >> 8;
|
||||||
|
}
|
||||||
|
msleep(200);
|
||||||
|
}
|
||||||
|
if (user_cancelled) {
|
||||||
|
gfx_printf("\nAbgebrochen. Starte Hekate...\n");
|
||||||
|
msleep(500);
|
||||||
|
if (file_exists(PAYLOAD_PATH)) {
|
||||||
|
launch_payload(PAYLOAD_PATH);
|
||||||
|
} else {
|
||||||
|
power_set_state(POWER_OFF_REBOOT);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Charger detected or battery OK - clear the low-battery screen before continuing
|
||||||
|
gfx_clear_grey(0x1B);
|
||||||
|
gfx_con_setpos(0, 0);
|
||||||
|
print_header();
|
||||||
|
|
||||||
// Show information
|
// Show information
|
||||||
set_color(COLOR_CYAN);
|
set_color(COLOR_CYAN);
|
||||||
gfx_printf("Installationsmodus: %s\n", mode == INSTALL_MODE_UPDATE ? "Update" : "Saubere Installation");
|
gfx_printf("Installationsmodus: %s\n", mode == INSTALL_MODE_UPDATE ? "Update" : "Saubere Installation");
|
||||||
@@ -400,33 +452,58 @@ void ipl_main(void) {
|
|||||||
gfx_printf("Druecke A-Taste (rechter Joy-Con) oder Power-Taste,\n");
|
gfx_printf("Druecke A-Taste (rechter Joy-Con) oder Power-Taste,\n");
|
||||||
gfx_printf("um die Installation zu starten...\n");
|
gfx_printf("um die Installation zu starten...\n");
|
||||||
set_color(COLOR_WHITE);
|
set_color(COLOR_WHITE);
|
||||||
|
set_color(COLOR_CYAN);
|
||||||
|
gfx_printf("Druecke + und - gleichzeitig zum Abbrechen (zurueck zu Hekate).\n");
|
||||||
|
set_color(COLOR_WHITE);
|
||||||
|
|
||||||
// Wait for either A button or Power button
|
// Wait for A/Power to start, or +/- to cancel
|
||||||
bool button_pressed = false;
|
bool button_pressed = false;
|
||||||
|
bool cancelled = false;
|
||||||
|
|
||||||
// First, wait for power button to be released if it's currently pressed
|
// First, wait for power button to be released if it's currently pressed
|
||||||
while (btn_read() & BTN_POWER) {
|
while (btn_read() & BTN_POWER) {
|
||||||
msleep(50);
|
msleep(50);
|
||||||
}
|
}
|
||||||
|
|
||||||
while (!button_pressed) {
|
while (!button_pressed && !cancelled) {
|
||||||
// Check power button - detect press (transition from not pressed to pressed)
|
// Check power button
|
||||||
u8 btn_state = btn_read();
|
u8 btn_state = btn_read();
|
||||||
if (btn_state & BTN_POWER) {
|
if (btn_state & BTN_POWER) {
|
||||||
button_pressed = true;
|
button_pressed = true;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check joycon A button
|
// Check joycon buttons
|
||||||
jc_gamepad_rpt_t *jc = joycon_poll();
|
jc_gamepad_rpt_t *jc = joycon_poll();
|
||||||
if (jc && jc->a) {
|
if (jc) {
|
||||||
button_pressed = true;
|
if (jc->a) {
|
||||||
break;
|
button_pressed = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
// + and - simultaneously = cancel, return to hekate
|
||||||
|
if (jc->plus && jc->minus) {
|
||||||
|
cancelled = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
msleep(50); // Small delay to avoid busy-waiting
|
msleep(50); // Small delay to avoid busy-waiting
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (cancelled) {
|
||||||
|
gfx_printf("\n");
|
||||||
|
set_color(COLOR_YELLOW);
|
||||||
|
gfx_printf("Abgebrochen. Starte Hekate...\n");
|
||||||
|
set_color(COLOR_WHITE);
|
||||||
|
msleep(500);
|
||||||
|
if (file_exists(PAYLOAD_PATH)) {
|
||||||
|
launch_payload(PAYLOAD_PATH);
|
||||||
|
} else {
|
||||||
|
power_set_state(POWER_OFF_REBOOT);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Clear the prompt and start installation
|
// Clear the prompt and start installation
|
||||||
gfx_clear_grey(0x1B);
|
gfx_clear_grey(0x1B);
|
||||||
gfx_con_setpos(0, 0);
|
gfx_con_setpos(0, 0);
|
||||||
|
|||||||
Reference in New Issue
Block a user