From b6310d3fd619826b7bacc10a8a7ceea99b04b915 Mon Sep 17 00:00:00 2001 From: schellingb <14200249+schellingb@users.noreply.github.com> Date: Thu, 17 Oct 2024 22:10:47 +0900 Subject: [PATCH] Avoid the frontend freezing for a long time while caching the ZIP seek cache the first time by showing a loading message (#519) Caching can take a minute or longer on a low end device, now the frontend stays responsive throughout the process. --- dosbox_pure_libretro.cpp | 25 +++++++++++++++++++++++++ src/dos/drive_zip.cpp | 25 +++++++++++++++++++++---- 2 files changed, 46 insertions(+), 4 deletions(-) diff --git a/dosbox_pure_libretro.cpp b/dosbox_pure_libretro.cpp index 7524255e..090502b3 100644 --- a/dosbox_pure_libretro.cpp +++ b/dosbox_pure_libretro.cpp @@ -2417,6 +2417,31 @@ void GFX_ShowMsg(char const* format,...) void GFX_SetPalette(Bitu start,Bitu count,GFX_PalEntry * entries) { } +void DBP_ShowSlowLoading() +{ + if (!render.src.fps || !render.src.width) return; + + // Force screen refresh without advancing the emulation (call GFX_StartUpdate to make sure buf is allocated) + dbp_framecount++; + buffer_active = (buffer_active + (3-1)) % 3; // go back + Bit8u* pixels; Bitu pitch; GFX_StartUpdate(pixels, pitch); + buffer_active = (buffer_active + 1) % 3; // advance again + DBP_BufferDrawing& buf = (DBP_BufferDrawing&)dbp_buffers[buffer_active]; + + // Show loading message + if (DBP_Run::autoinput.ptr) memset(buf.video, 0, buf.width * buf.height * 4); // keep black during auto input + buf.DrawBox(20, buf.height - 40, buf.width - 40, 20, buf.BGCOL_MENU | 0x80000000, buf.COL_LINEBOX); + char msg[24]; + strcpy(msg, "Caching ZIP Structure "); + static const char charanim[] = { '|', '/', '-', '\\' }; + msg[22] = charanim[dbp_framecount % 4]; + buf.PrintCenteredOutlined(14, 0, buf.width, buf.height - 37, msg, buf.COL_MENUTITLE, 0x80404020); + + // Prevent main thread from mixing audio because we could be called while inside MixerChannel::Mix + dbp_audio_remain = -1; + GFX_Events_AdvanceFrame(false); +} + static void DBP_PureLabelProgram(Program** make) { struct LabelProgram : Program diff --git a/src/dos/drive_zip.cpp b/src/dos/drive_zip.cpp index 43f98b62..0d1e7d8d 100644 --- a/src/dos/drive_zip.cpp +++ b/src/dos/drive_zip.cpp @@ -1,5 +1,5 @@ /* - * Copyright (C) 2020-2023 Bernhard Schelling + * Copyright (C) 2020-2024 Bernhard Schelling * * 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 @@ -1377,9 +1377,8 @@ struct Zip_DeflateUnpacker : ZIP_Unpacker Bit32u Read(const Zip_File& f, Bit32u seek_ofs, void *res_buf, Bit32u res_n) { if (crc_failed) return 0; - Bit32u want_from = seek_ofs, want_to = seek_ofs + res_n; + Bit32u want_from = seek_ofs, want_to = seek_ofs + res_n, last_idx = (Bit32u)-1, slowload_num, slowload_tick; DBP_ASSERT(want_to <= f.decomp_size); - Bit8u* p_res = (Bit8u*)res_buf; Bit32u have_from = ((out_buf_ofs ? out_buf_ofs - 1 : 0) & ~(WRITE_BLOCK-1)); if (want_from < have_from || want_from > out_buf_ofs) @@ -1411,6 +1410,7 @@ struct Zip_DeflateUnpacker : ZIP_Unpacker } } + Bit8u* p_res = (Bit8u*)res_buf; for (miniz::tinfl_status status = miniz::TINFL_STATUS_NEEDS_MORE_INPUT; status == miniz::TINFL_STATUS_NEEDS_MORE_INPUT || status == miniz::TINFL_STATUS_HAS_MORE_OUTPUT || status == miniz::TINFL_STATUS_DONE;) { if (out_buf_ofs > want_from) @@ -1462,7 +1462,7 @@ struct Zip_DeflateUnpacker : ZIP_Unpacker if (inflator.m_state == miniz::TINFL_STATE_INDEX_BLOCK_BOUNDRY) { - // Gear cursors toward the middle of the block to accomodate forward and backward seeking as well as possible + // Gear cursors toward the middle of the block to accommodate forward and backward seeking as well as possible Bit32u idx = (out_buf_ofs / cursor_block); if (!cursors[idx].cursor_out || (out_buf_ofs > cursors[idx].cursor_out + 120*1024 && out_buf_ofs < idx*cursor_block + cursor_block/2 + 70*1024)) { @@ -1514,6 +1514,23 @@ struct Zip_DeflateUnpacker : ZIP_Unpacker seek_cache->cache_count = cursor_got; } } + + if (last_idx != idx && seek_cache && idx > 50) + { + extern Bit32u DBP_GetTicks(); + Bit32u tick = DBP_GetTicks(); + if (last_idx == (Bit32u)-1) { slowload_num = 0; slowload_tick = tick + 77; } + else if (slowload_num++ >= SEEK_CACHE_CURSOR_STEPS*2 && (Bit32s)(tick - slowload_tick) >= 33) + { + slowload_tick = tick; + extern void DBP_ShowSlowLoading(); + DBP_ShowSlowLoading(); + extern bool DBP_IsShuttingDown(); + if (DBP_IsShuttingDown()) + return (Bit32u)(p_res - (Bit8u*)res_buf); + } + last_idx = idx; + } } } }