2018-09-07 16:00:13 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018 Atmosphère-NX
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms and conditions of the GNU General Public License,
|
|
|
|
* version 2, as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
2018-09-28 21:12:59 +01:00
|
|
|
|
|
|
|
#include "console.h"
|
2018-09-17 21:03:03 +01:00
|
|
|
#include "di.h"
|
2018-04-10 16:37:58 -06:00
|
|
|
#include "timers.h"
|
2018-04-10 16:09:00 -06:00
|
|
|
#include "splash_screen.h"
|
2018-05-13 23:49:50 +02:00
|
|
|
#include "fs_utils.h"
|
2018-04-10 16:09:00 -06:00
|
|
|
#include "display/video_fb.h"
|
|
|
|
|
2018-09-17 21:03:03 +01:00
|
|
|
#define u8 uint8_t
|
|
|
|
#define u32 uint32_t
|
|
|
|
#include "splash_screen_bmp.h"
|
|
|
|
#undef u8
|
|
|
|
#undef u32
|
|
|
|
|
2019-01-26 00:59:27 -08:00
|
|
|
static uint32_t g_splash_start_time = 0;
|
|
|
|
|
2018-09-17 21:03:03 +01:00
|
|
|
static void render_bmp(const uint32_t *bmp_data, uint32_t *framebuffer, uint32_t bmp_width, uint32_t bmp_height, uint32_t bmp_pos_x, uint32_t bmp_pos_y) {
|
|
|
|
/* Render the BMP. */
|
|
|
|
for (uint32_t y = bmp_pos_y; y < (bmp_pos_y + bmp_height); y++) {
|
|
|
|
for (uint32_t x = bmp_pos_x; x < (bmp_pos_x + bmp_width); x++) {
|
|
|
|
framebuffer[x + (y * SPLASH_SCREEN_STRIDE)] = bmp_data[(bmp_height + bmp_pos_y - 1 - y) * bmp_width + x - bmp_pos_x];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Re-initialize the frame buffer. */
|
2018-09-28 21:12:59 +01:00
|
|
|
console_display(framebuffer);
|
2018-09-17 21:03:03 +01:00
|
|
|
}
|
|
|
|
|
2019-01-26 00:59:27 -08:00
|
|
|
void splash_screen_wait_delay(void) {
|
|
|
|
/* Ensure the splash screen is displayed for at least three seconds. */
|
|
|
|
udelay_absolute(g_splash_start_time, 3000000);
|
|
|
|
}
|
|
|
|
|
2018-09-17 21:03:03 +01:00
|
|
|
void display_splash_screen_bmp(const char *custom_splash_path, void *fb_address) {
|
|
|
|
uint8_t *splash_screen = (uint8_t *)splash_screen_bmp;
|
|
|
|
|
|
|
|
/* Try to load an external custom splash screen. */
|
2018-08-31 22:46:08 +00:00
|
|
|
if ((custom_splash_path != NULL) && (custom_splash_path[0] != '\x00')) {
|
2018-11-19 20:05:47 +00:00
|
|
|
if (!read_from_file(splash_screen, splash_screen_bmp_size, custom_splash_path)) {
|
2018-05-20 16:18:48 +02:00
|
|
|
fatal_error("Failed to read custom splash screen from %s!\n", custom_splash_path);
|
2018-04-10 16:09:00 -06:00
|
|
|
}
|
|
|
|
}
|
2018-09-17 21:03:03 +01:00
|
|
|
|
|
|
|
/* Check for 'BM' magic. */
|
|
|
|
if ((splash_screen[0] == 'B') && (splash_screen[1] == 'M')) {
|
|
|
|
/* Extract BMP parameters. */
|
|
|
|
uint32_t bmp_size = (splash_screen[0x02] | (splash_screen[0x03] << 8) | (splash_screen[0x04] << 16) | (splash_screen[0x05] << 24));
|
|
|
|
uint32_t bmp_offset = (splash_screen[0x0A] | (splash_screen[0x0B] << 8) | (splash_screen[0x0C] << 16) | (splash_screen[0x0D] << 24));
|
|
|
|
uint32_t bmp_width = (splash_screen[0x12] | (splash_screen[0x13] << 8) | (splash_screen[0x14] << 16) | (splash_screen[0x15] << 24));
|
|
|
|
uint32_t bmp_height = (splash_screen[0x16] | (splash_screen[0x17] << 8) | (splash_screen[0x18] << 16) | (splash_screen[0x19] << 24));
|
|
|
|
uint16_t bmp_bpp = (splash_screen[0x1C] | (splash_screen[0x1D] << 8));
|
|
|
|
uint32_t bmp_data_size = (splash_screen[0x22] | (splash_screen[0x23] << 8) | (splash_screen[0x24] << 16) | (splash_screen[0x25] << 24));
|
|
|
|
|
|
|
|
/* Data size can be wrong or set to 0. In that case, we calculate it instead. */
|
|
|
|
if (!bmp_data_size || (bmp_data_size >= bmp_size))
|
|
|
|
bmp_data_size = (bmp_size - bmp_offset);
|
|
|
|
|
|
|
|
/* Only accept images up to 720x1280 resolution and with 32 BPP. */
|
|
|
|
if ((bmp_width > SPLASH_SCREEN_WIDTH_MAX) || (bmp_height > SPLASH_SCREEN_HEIGHT_MAX)) {
|
|
|
|
fatal_error("Invalid splash screen dimensions!\n");
|
|
|
|
} else if (bmp_bpp != SPLASH_SCREEN_BPP) {
|
|
|
|
fatal_error("Invalid splash screen color depth!\n");
|
|
|
|
} else if (bmp_data_size > SPLASH_SCREEN_SIZE_MAX) {
|
|
|
|
fatal_error("Splash screen data size is too big!\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Calculate screen positions. */
|
|
|
|
uint32_t bmp_pos_x = ((SPLASH_SCREEN_WIDTH_MAX - bmp_width) / 2);
|
|
|
|
uint32_t bmp_pos_y = ((SPLASH_SCREEN_HEIGHT_MAX - bmp_height) / 2);
|
|
|
|
|
|
|
|
/* Advance to data. */
|
|
|
|
splash_screen += bmp_offset;
|
|
|
|
|
|
|
|
/* Render the BMP. */
|
|
|
|
render_bmp((uint32_t *)splash_screen, (uint32_t *)fb_address, bmp_width, bmp_height, bmp_pos_x, bmp_pos_y);
|
|
|
|
} else {
|
|
|
|
fatal_error("Invalid splash screen format!\n");
|
|
|
|
}
|
2019-01-26 00:59:27 -08:00
|
|
|
|
|
|
|
/* Note the time we started displaying the splash. */
|
|
|
|
g_splash_start_time = get_time_us();
|
2018-05-05 17:33:49 +02:00
|
|
|
}
|