mirror of
https://github.com/trezor/trezor-firmware.git
synced 2025-01-22 21:30:56 +00:00
refactor(core): improve safety of buffer handling
[no changelog]
This commit is contained in:
parent
640685a41f
commit
69be9b1edf
@ -22,82 +22,61 @@
|
|||||||
#include "fonts/fonts.h"
|
#include "fonts/fonts.h"
|
||||||
#include "memzero.h"
|
#include "memzero.h"
|
||||||
|
|
||||||
#define BUFFERS_16BPP 3
|
|
||||||
#define BUFFERS_4BPP 3
|
|
||||||
#define BUFFERS_TEXT 1
|
|
||||||
#define BUFFERS_JPEG 1
|
|
||||||
#define BUFFERS_JPEG_WORK 1
|
|
||||||
#define BUFFERS_BLURRING 1
|
|
||||||
|
|
||||||
const int32_t text_buffer_height = FONT_MAX_HEIGHT;
|
const int32_t text_buffer_height = FONT_MAX_HEIGHT;
|
||||||
const int32_t buffer_width = DISPLAY_RESX;
|
const int32_t buffer_width = DISPLAY_RESX;
|
||||||
|
|
||||||
BUFFER_SECTION line_buffer_16bpp_t line_buffers_16bpp[BUFFERS_16BPP];
|
#define CONCAT_(a, b) a##b
|
||||||
BUFFER_SECTION line_buffer_4bpp_t line_buffers_4bpp[BUFFERS_4BPP];
|
#define CONCAT(a, b) CONCAT_(a, b)
|
||||||
BUFFER_SECTION buffer_text_t text_buffers[BUFFERS_TEXT];
|
|
||||||
NODMA_BUFFER_SECTION buffer_jpeg_t jpeg_buffers[BUFFERS_JPEG];
|
|
||||||
NODMA_BUFFER_SECTION buffer_jpeg_work_t jpeg_work_buffers[BUFFERS_JPEG_WORK];
|
|
||||||
NODMA_BUFFER_SECTION buffer_blurring_t blurring_buffers[BUFFERS_BLURRING];
|
|
||||||
|
|
||||||
line_buffer_16bpp_t* buffers_get_line_buffer_16bpp(uint16_t idx, bool clear) {
|
#define CONCAT3_(a, b, c) a##b##c
|
||||||
if (idx >= BUFFERS_16BPP) {
|
#define CONCAT3(a, b, c) CONCAT3_(a, b, c)
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
if (clear) {
|
|
||||||
memzero(&line_buffers_16bpp[idx], sizeof(line_buffers_16bpp[idx]));
|
|
||||||
}
|
|
||||||
return &line_buffers_16bpp[idx];
|
|
||||||
}
|
|
||||||
|
|
||||||
line_buffer_4bpp_t* buffers_get_line_buffer_4bpp(uint16_t idx, bool clear) {
|
#define STRUCT(name) CONCAT3(buffers_, name, _t)
|
||||||
if (idx >= BUFFERS_4BPP) {
|
#define TYPE(name) CONCAT3(buffer_, name, _t)
|
||||||
return NULL;
|
#define FUNCTION(name) CONCAT(buffers_get_, name)
|
||||||
}
|
#define FUNCTION_FREE(name) CONCAT(buffers_free_, name)
|
||||||
if (clear) {
|
#define VARNAME(name) CONCAT(buffers_, name)
|
||||||
memzero(&line_buffers_4bpp[idx], sizeof(line_buffers_4bpp[idx]));
|
|
||||||
}
|
|
||||||
return &line_buffers_4bpp[idx];
|
|
||||||
}
|
|
||||||
|
|
||||||
buffer_text_t* buffers_get_text_buffer(uint16_t idx, bool clear) {
|
#define BUFFER(section, name, count) \
|
||||||
if (idx >= BUFFERS_TEXT) {
|
typedef struct { \
|
||||||
return NULL;
|
TYPE(name) buffers[count]; \
|
||||||
}
|
uint8_t allocated[count]; \
|
||||||
if (clear) {
|
} STRUCT(name); \
|
||||||
memzero(&text_buffers[idx], sizeof(text_buffers[idx]));
|
section STRUCT(name) VARNAME(name); \
|
||||||
}
|
\
|
||||||
return &text_buffers[idx];
|
TYPE(name) * FUNCTION(name)(bool clear) { \
|
||||||
}
|
int idx = -1; \
|
||||||
|
for (int i = 0; i < (count); i++) { \
|
||||||
buffer_jpeg_t* buffers_get_jpeg_buffer(uint16_t idx, bool clear) {
|
if (VARNAME(name).allocated[i] == 0) { \
|
||||||
if (idx >= BUFFERS_JPEG) {
|
idx = i; \
|
||||||
return NULL;
|
break; \
|
||||||
|
} \
|
||||||
|
} \
|
||||||
|
if (idx < 0) { \
|
||||||
|
return NULL; \
|
||||||
|
} \
|
||||||
|
if (clear) { \
|
||||||
|
memzero(&VARNAME(name).buffers[idx], \
|
||||||
|
sizeof(VARNAME(name).buffers[idx])); \
|
||||||
|
} \
|
||||||
|
VARNAME(name).allocated[idx] = 1; \
|
||||||
|
return &VARNAME(name).buffers[idx]; \
|
||||||
|
} \
|
||||||
|
void FUNCTION_FREE(name)(TYPE(name) * buffer) { \
|
||||||
|
if (buffer == NULL) { \
|
||||||
|
return; \
|
||||||
|
} \
|
||||||
|
for (uint16_t i = 0; i < (count); i++) { \
|
||||||
|
if (buffer == &VARNAME(name).buffers[i]) { \
|
||||||
|
VARNAME(name).allocated[i] = 0; \
|
||||||
|
return; \
|
||||||
|
} \
|
||||||
|
} \
|
||||||
}
|
}
|
||||||
|
|
||||||
if (clear) {
|
BUFFER(BUFFER_SECTION, line_16bpp, 3);
|
||||||
memzero(&jpeg_buffers[idx], sizeof(jpeg_buffers[idx]));
|
BUFFER(BUFFER_SECTION, line_4bpp, 3);
|
||||||
}
|
BUFFER(BUFFER_SECTION, text, 1);
|
||||||
return &jpeg_buffers[idx];
|
BUFFER(NODMA_BUFFER_SECTION, jpeg, 1);
|
||||||
}
|
BUFFER(NODMA_BUFFER_SECTION, jpeg_work, 1);
|
||||||
|
BUFFER(NODMA_BUFFER_SECTION, blurring, 1);
|
||||||
buffer_jpeg_work_t* buffers_get_jpeg_work_buffer(uint16_t idx, bool clear) {
|
|
||||||
if (idx >= BUFFERS_JPEG_WORK) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (clear) {
|
|
||||||
memzero(&jpeg_work_buffers[idx], sizeof(jpeg_work_buffers[idx]));
|
|
||||||
}
|
|
||||||
return &jpeg_work_buffers[idx];
|
|
||||||
}
|
|
||||||
|
|
||||||
buffer_blurring_t* buffers_get_blurring_buffer(uint16_t idx, bool clear) {
|
|
||||||
if (idx >= BUFFERS_BLURRING) {
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (clear) {
|
|
||||||
memzero(&blurring_buffers[idx], sizeof(blurring_buffers[idx]));
|
|
||||||
}
|
|
||||||
return &blurring_buffers[idx];
|
|
||||||
}
|
|
||||||
|
@ -59,11 +59,11 @@
|
|||||||
|
|
||||||
typedef __attribute__((aligned(4))) struct {
|
typedef __attribute__((aligned(4))) struct {
|
||||||
uint8_t buffer[LINE_BUFFER_16BPP_SIZE];
|
uint8_t buffer[LINE_BUFFER_16BPP_SIZE];
|
||||||
} line_buffer_16bpp_t;
|
} buffer_line_16bpp_t;
|
||||||
|
|
||||||
typedef __attribute__((aligned(4))) struct {
|
typedef __attribute__((aligned(4))) struct {
|
||||||
uint8_t buffer[LINE_BUFFER_4BPP_SIZE];
|
uint8_t buffer[LINE_BUFFER_4BPP_SIZE];
|
||||||
} line_buffer_4bpp_t;
|
} buffer_line_4bpp_t;
|
||||||
|
|
||||||
typedef __attribute__((aligned(4))) struct {
|
typedef __attribute__((aligned(4))) struct {
|
||||||
uint8_t buffer[TEXT_BUFFER_SIZE];
|
uint8_t buffer[TEXT_BUFFER_SIZE];
|
||||||
@ -84,11 +84,22 @@ typedef __attribute__((aligned(4))) struct {
|
|||||||
extern const int32_t text_buffer_height;
|
extern const int32_t text_buffer_height;
|
||||||
extern const int32_t buffer_width;
|
extern const int32_t buffer_width;
|
||||||
|
|
||||||
line_buffer_16bpp_t* buffers_get_line_buffer_16bpp(uint16_t idx, bool clear);
|
buffer_line_16bpp_t* buffers_get_line_16bpp(bool clear);
|
||||||
line_buffer_4bpp_t* buffers_get_line_buffer_4bpp(uint16_t idx, bool clear);
|
void buffers_free_line_16bpp(buffer_line_16bpp_t* buffer);
|
||||||
buffer_text_t* buffers_get_text_buffer(uint16_t idx, bool clear);
|
|
||||||
buffer_jpeg_t* buffers_get_jpeg_buffer(uint16_t idx, bool clear);
|
buffer_line_4bpp_t* buffers_get_line_4bpp(bool clear);
|
||||||
buffer_jpeg_work_t* buffers_get_jpeg_work_buffer(uint16_t idx, bool clear);
|
void buffers_free_line_4bpp(buffer_line_4bpp_t* buffer);
|
||||||
buffer_blurring_t* buffers_get_blurring_buffer(uint16_t idx, bool clear);
|
|
||||||
|
buffer_text_t* buffers_get_text(bool clear);
|
||||||
|
void buffers_free_text(buffer_text_t* buffer);
|
||||||
|
|
||||||
|
buffer_jpeg_t* buffers_get_jpeg(bool clear);
|
||||||
|
void buffers_free_jpeg(buffer_jpeg_t* buffer);
|
||||||
|
|
||||||
|
buffer_jpeg_work_t* buffers_get_jpeg_work(bool clear);
|
||||||
|
void buffers_free_jpeg_work(buffer_jpeg_work_t* buffer);
|
||||||
|
|
||||||
|
buffer_blurring_t* buffers_get_blurring(bool clear);
|
||||||
|
void buffers_free_blurring(buffer_blurring_t* buffer);
|
||||||
|
|
||||||
#endif // _BUFFERS_H
|
#endif // _BUFFERS_H
|
||||||
|
@ -312,8 +312,13 @@ void display_image(int x, int y, int w, int h, const void *data,
|
|||||||
struct uzlib_uncomp decomp = {0};
|
struct uzlib_uncomp decomp = {0};
|
||||||
uint8_t decomp_window[UZLIB_WINDOW_SIZE] = {0};
|
uint8_t decomp_window[UZLIB_WINDOW_SIZE] = {0};
|
||||||
|
|
||||||
line_buffer_16bpp_t *b1 = buffers_get_line_buffer_16bpp(0, false);
|
buffer_line_16bpp_t *b1 = buffers_get_line_16bpp(false);
|
||||||
line_buffer_16bpp_t *b2 = buffers_get_line_buffer_16bpp(1, false);
|
if (b1 == NULL) return;
|
||||||
|
buffer_line_16bpp_t *b2 = buffers_get_line_16bpp(false);
|
||||||
|
if (b2 == NULL) {
|
||||||
|
buffers_free_line_16bpp(b1);
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
uzlib_prepare(&decomp, decomp_window, data, datalen, b1, w * 2);
|
uzlib_prepare(&decomp, decomp_window, data, datalen, b1, w * 2);
|
||||||
|
|
||||||
@ -321,7 +326,7 @@ void display_image(int x, int y, int w, int h, const void *data,
|
|||||||
|
|
||||||
for (int32_t pos = 0; pos < h; pos++) {
|
for (int32_t pos = 0; pos < h; pos++) {
|
||||||
int32_t pixels = w;
|
int32_t pixels = w;
|
||||||
line_buffer_16bpp_t *next_buf = (pos % 2 == 1) ? b1 : b2;
|
buffer_line_16bpp_t *next_buf = (pos % 2 == 1) ? b1 : b2;
|
||||||
decomp.dest = next_buf->buffer;
|
decomp.dest = next_buf->buffer;
|
||||||
decomp.dest_limit = next_buf->buffer + w * 2;
|
decomp.dest_limit = next_buf->buffer + w * 2;
|
||||||
int st = uzlib_uncompress(&decomp);
|
int st = uzlib_uncompress(&decomp);
|
||||||
@ -330,6 +335,8 @@ void display_image(int x, int y, int w, int h, const void *data,
|
|||||||
dma2d_start(next_buf->buffer, (uint8_t *)DISPLAY_DATA_ADDRESS, pixels);
|
dma2d_start(next_buf->buffer, (uint8_t *)DISPLAY_DATA_ADDRESS, pixels);
|
||||||
}
|
}
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
|
buffers_free_line_16bpp(b1);
|
||||||
|
buffers_free_line_16bpp(b2);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -391,8 +398,13 @@ void display_icon(int x, int y, int w, int h, const void *data,
|
|||||||
}
|
}
|
||||||
|
|
||||||
uint8_t b[DISPLAY_RESX / 2];
|
uint8_t b[DISPLAY_RESX / 2];
|
||||||
line_buffer_4bpp_t *b1 = buffers_get_line_buffer_4bpp(0, false);
|
buffer_line_4bpp_t *b1 = buffers_get_line_4bpp(false);
|
||||||
line_buffer_4bpp_t *b2 = buffers_get_line_buffer_4bpp(1, false);
|
if (b1 == NULL) return;
|
||||||
|
buffer_line_4bpp_t *b2 = buffers_get_line_4bpp(false);
|
||||||
|
if (b2 == NULL) {
|
||||||
|
buffers_free_line_4bpp(b1);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
struct uzlib_uncomp decomp = {0};
|
struct uzlib_uncomp decomp = {0};
|
||||||
uint8_t decomp_window[UZLIB_WINDOW_SIZE] = {0};
|
uint8_t decomp_window[UZLIB_WINDOW_SIZE] = {0};
|
||||||
@ -404,7 +416,7 @@ void display_icon(int x, int y, int w, int h, const void *data,
|
|||||||
int off_x = x < 0 ? -x : 0;
|
int off_x = x < 0 ? -x : 0;
|
||||||
|
|
||||||
for (uint32_t pos = 0; pos < h; pos++) {
|
for (uint32_t pos = 0; pos < h; pos++) {
|
||||||
line_buffer_4bpp_t *next_buf = (pos % 2 == 0) ? b1 : b2;
|
buffer_line_4bpp_t *next_buf = (pos % 2 == 0) ? b1 : b2;
|
||||||
decomp.dest = b;
|
decomp.dest = b;
|
||||||
decomp.dest_limit = b + w / 2;
|
decomp.dest_limit = b + w / 2;
|
||||||
int st = uzlib_uncompress(&decomp);
|
int st = uzlib_uncompress(&decomp);
|
||||||
@ -416,6 +428,8 @@ void display_icon(int x, int y, int w, int h, const void *data,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
|
buffers_free_line_4bpp(b1);
|
||||||
|
buffers_free_line_4bpp(b2);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -328,12 +328,24 @@ fn generate_trezorhal_bindings() {
|
|||||||
.allowlist_function("dma2d_start_blend")
|
.allowlist_function("dma2d_start_blend")
|
||||||
.allowlist_function("dma2d_wait_for_transfer")
|
.allowlist_function("dma2d_wait_for_transfer")
|
||||||
//buffers
|
//buffers
|
||||||
.allowlist_function("buffers_get_line_buffer_16bpp")
|
.allowlist_function("buffers_get_line_16bpp")
|
||||||
.allowlist_function("buffers_get_line_buffer_4bpp")
|
.allowlist_function("buffers_free_line_16bpp")
|
||||||
.allowlist_function("buffers_get_text_buffer")
|
.allowlist_function("buffers_get_line_4bpp")
|
||||||
.allowlist_function("buffers_get_jpeg_buffer")
|
.allowlist_function("buffers_free_line_4bpp")
|
||||||
.allowlist_function("buffers_get_jpeg_work_buffer")
|
.allowlist_function("buffers_get_text")
|
||||||
.allowlist_function("buffers_get_blurring_buffer")
|
.allowlist_function("buffers_free_text")
|
||||||
|
.allowlist_function("buffers_get_jpeg")
|
||||||
|
.allowlist_function("buffers_free_jpeg")
|
||||||
|
.allowlist_function("buffers_get_jpeg_work")
|
||||||
|
.allowlist_function("buffers_free_jpeg_work")
|
||||||
|
.allowlist_function("buffers_get_blurring")
|
||||||
|
.allowlist_function("buffers_free_blurring")
|
||||||
|
.no_copy("buffer_line_16bpp_t")
|
||||||
|
.no_copy("buffer_line_4bpp_t")
|
||||||
|
.no_copy("buffer_text_t")
|
||||||
|
.no_copy("buffer_jpeg_t")
|
||||||
|
.no_copy("buffer_jpeg_work_t")
|
||||||
|
.no_copy("buffer_blurring_t")
|
||||||
.allowlist_var("text_buffer_height")
|
.allowlist_var("text_buffer_height")
|
||||||
.allowlist_var("buffer_width")
|
.allowlist_var("buffer_width")
|
||||||
//usb
|
//usb
|
||||||
|
@ -1,88 +1,93 @@
|
|||||||
|
use core::{
|
||||||
|
ops::{Deref, DerefMut},
|
||||||
|
ptr,
|
||||||
|
};
|
||||||
|
|
||||||
use super::ffi;
|
use super::ffi;
|
||||||
|
|
||||||
pub use ffi::{
|
macro_rules! buffer_wrapper {
|
||||||
buffer_blurring_t as BlurringBuffer, buffer_text_t as BufferText,
|
($rust_name: ident, $type: ident, $get: ident, $free: ident) => {
|
||||||
line_buffer_16bpp_t as LineBuffer16Bpp, line_buffer_4bpp_t as LineBuffer4Bpp,
|
pub struct $rust_name(ptr::NonNull<ffi::$type>);
|
||||||
};
|
|
||||||
|
impl $rust_name {
|
||||||
|
pub fn get() -> Self {
|
||||||
|
unsafe {
|
||||||
|
let ptr = ffi::$get(false);
|
||||||
|
Self(unwrap!(ptr::NonNull::new(ptr)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_cleared() -> Self {
|
||||||
|
unsafe {
|
||||||
|
let ptr = ffi::$get(true);
|
||||||
|
Self(unwrap!(ptr::NonNull::new(ptr)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Deref for $rust_name {
|
||||||
|
type Target = ffi::$type;
|
||||||
|
|
||||||
|
fn deref(&self) -> &Self::Target {
|
||||||
|
// SAFETY: The lifetime of the pointer is 'static and the C API
|
||||||
|
// promises that we are the sole owner.
|
||||||
|
unsafe { self.0.as_ref() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl DerefMut for $rust_name {
|
||||||
|
fn deref_mut(&mut self) -> &mut Self::Target {
|
||||||
|
// SAFETY: The lifetime of the pointer is 'static and the C API
|
||||||
|
// promises that we are the sole owner, and we have borrowed mutably.
|
||||||
|
unsafe { self.0.as_mut() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for $rust_name {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
unsafe {
|
||||||
|
ffi::$free(self.0.as_mut());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
buffer_wrapper!(
|
||||||
|
BufferLine16bpp,
|
||||||
|
buffer_line_16bpp_t,
|
||||||
|
buffers_get_line_16bpp,
|
||||||
|
buffers_free_line_16bpp
|
||||||
|
);
|
||||||
|
buffer_wrapper!(
|
||||||
|
BufferLine4bpp,
|
||||||
|
buffer_line_4bpp_t,
|
||||||
|
buffers_get_line_4bpp,
|
||||||
|
buffers_free_line_4bpp
|
||||||
|
);
|
||||||
|
buffer_wrapper!(
|
||||||
|
BufferText,
|
||||||
|
buffer_text_t,
|
||||||
|
buffers_get_text,
|
||||||
|
buffers_free_text
|
||||||
|
);
|
||||||
|
buffer_wrapper!(
|
||||||
|
BufferBlurring,
|
||||||
|
buffer_blurring_t,
|
||||||
|
buffers_get_blurring,
|
||||||
|
buffers_free_blurring
|
||||||
|
);
|
||||||
#[cfg(feature = "jpeg")]
|
#[cfg(feature = "jpeg")]
|
||||||
pub use ffi::{buffer_jpeg_t as BufferJpeg, buffer_jpeg_work_t as BufferJpegWork};
|
buffer_wrapper!(
|
||||||
|
BufferJpeg,
|
||||||
/// Returns a buffer for one line of 16bpp data
|
buffer_jpeg_t,
|
||||||
///
|
buffers_get_jpeg,
|
||||||
/// # Safety
|
buffers_free_jpeg
|
||||||
///
|
);
|
||||||
/// This function is unsafe because the caller has to guarantee
|
|
||||||
/// that he doesn't use buffer on same index multiple times
|
|
||||||
pub unsafe fn get_buffer_16bpp(idx: u16, clear: bool) -> &'static mut LineBuffer16Bpp {
|
|
||||||
unsafe {
|
|
||||||
let ptr = ffi::buffers_get_line_buffer_16bpp(idx, clear);
|
|
||||||
unwrap!(ptr.as_mut())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a buffer for one line of 4bpp data
|
|
||||||
///
|
|
||||||
/// # Safety
|
|
||||||
///
|
|
||||||
/// This function is unsafe because the caller has to guarantee
|
|
||||||
/// that he doesn't use buffer on same index multiple times
|
|
||||||
pub unsafe fn get_buffer_4bpp(idx: u16, clear: bool) -> &'static mut LineBuffer4Bpp {
|
|
||||||
unsafe {
|
|
||||||
let ptr = ffi::buffers_get_line_buffer_4bpp(idx, clear);
|
|
||||||
unwrap!(ptr.as_mut())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a buffer for one line of text
|
|
||||||
///
|
|
||||||
/// # Safety
|
|
||||||
///
|
|
||||||
/// This function is unsafe because the caller has to guarantee
|
|
||||||
/// that he doesn't use buffer on same index multiple times
|
|
||||||
pub unsafe fn get_text_buffer(idx: u16, clear: bool) -> &'static mut BufferText {
|
|
||||||
unsafe {
|
|
||||||
let ptr = ffi::buffers_get_text_buffer(idx, clear);
|
|
||||||
unwrap!(ptr.as_mut())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a buffer for jpeg data
|
|
||||||
///
|
|
||||||
/// # Safety
|
|
||||||
///
|
|
||||||
/// This function is unsafe because the caller has to guarantee
|
|
||||||
/// that he doesn't use buffer on same index multiple times
|
|
||||||
#[cfg(feature = "jpeg")]
|
#[cfg(feature = "jpeg")]
|
||||||
pub unsafe fn get_jpeg_buffer(idx: u16, clear: bool) -> &'static mut BufferJpeg {
|
buffer_wrapper!(
|
||||||
unsafe {
|
BufferJpegWork,
|
||||||
let ptr = ffi::buffers_get_jpeg_buffer(idx, clear);
|
buffer_jpeg_work_t,
|
||||||
unwrap!(ptr.as_mut())
|
buffers_get_jpeg_work,
|
||||||
}
|
buffers_free_jpeg_work
|
||||||
}
|
);
|
||||||
|
|
||||||
/// Returns a jpeg work buffer
|
|
||||||
///
|
|
||||||
/// # Safety
|
|
||||||
///
|
|
||||||
/// This function is unsafe because the caller has to guarantee
|
|
||||||
/// that he doesn't use buffer on same index multiple times
|
|
||||||
#[cfg(feature = "jpeg")]
|
|
||||||
pub unsafe fn get_jpeg_work_buffer(idx: u16, clear: bool) -> &'static mut BufferJpegWork {
|
|
||||||
unsafe {
|
|
||||||
let ptr = ffi::buffers_get_jpeg_work_buffer(idx, clear);
|
|
||||||
unwrap!(ptr.as_mut())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Returns a buffer for blurring data
|
|
||||||
///
|
|
||||||
/// # Safety
|
|
||||||
///
|
|
||||||
/// This function is unsafe because the caller has to guarantee
|
|
||||||
/// that he doesn't use buffer on same index multiple times
|
|
||||||
pub unsafe fn get_blurring_buffer(idx: u16, clear: bool) -> &'static mut BlurringBuffer {
|
|
||||||
unsafe {
|
|
||||||
let ptr = ffi::buffers_get_blurring_buffer(idx, clear);
|
|
||||||
unwrap!(ptr.as_mut())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
use super::ffi;
|
use super::ffi;
|
||||||
use core::ptr;
|
use core::{ops::DerefMut, ptr};
|
||||||
use cty::c_int;
|
use cty::c_int;
|
||||||
|
|
||||||
use crate::trezorhal::buffers::BufferText;
|
use crate::trezorhal::buffers::BufferText;
|
||||||
@ -36,7 +36,7 @@ pub fn text_into_buffer(text: &str, font: i32, buffer: &mut BufferText, x_offset
|
|||||||
text.as_ptr() as _,
|
text.as_ptr() as _,
|
||||||
text.len() as _,
|
text.len() as _,
|
||||||
font,
|
font,
|
||||||
buffer as _,
|
buffer.deref_mut(),
|
||||||
x_offset.into(),
|
x_offset.into(),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@ -92,7 +92,14 @@ pub fn bar_radius(x: i16, y: i16, w: i16, h: i16, fgcolor: u16, bgcolor: u16, ra
|
|||||||
|
|
||||||
pub fn bar_radius_buffer(x: i16, y: i16, w: i16, h: i16, radius: u8, buffer: &mut BufferText) {
|
pub fn bar_radius_buffer(x: i16, y: i16, w: i16, h: i16, radius: u8, buffer: &mut BufferText) {
|
||||||
unsafe {
|
unsafe {
|
||||||
ffi::display_bar_radius_buffer(x.into(), y.into(), w.into(), h.into(), radius, buffer as _)
|
ffi::display_bar_radius_buffer(
|
||||||
|
x.into(),
|
||||||
|
y.into(),
|
||||||
|
w.into(),
|
||||||
|
h.into(),
|
||||||
|
radius,
|
||||||
|
buffer.deref_mut(),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -12,7 +12,16 @@ pub fn dma2d_setup_4bpp_over_16bpp(overlay_color: u16) {
|
|||||||
unsafe { ffi::dma2d_setup_4bpp_over_16bpp(overlay_color) }
|
unsafe { ffi::dma2d_setup_4bpp_over_16bpp(overlay_color) }
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn dma2d_start(buffer: &[u8], pixels: i16) {
|
/// Starts dma2d transfer
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// This function is unsafe because the caller has to guarantee that he:
|
||||||
|
/// 1) doesn't mutate the buffers until the transfer is finished, which is
|
||||||
|
/// guaranteed by calling `dma2d_wait_for_transfer`
|
||||||
|
/// 2) the buffer doesn't get dropped until the transfer is finished
|
||||||
|
/// 3) doesn't call this function while another transfer is running
|
||||||
|
pub unsafe fn dma2d_start(buffer: &[u8], pixels: i16) {
|
||||||
unsafe {
|
unsafe {
|
||||||
ffi::dma2d_start(
|
ffi::dma2d_start(
|
||||||
buffer.as_ptr() as _,
|
buffer.as_ptr() as _,
|
||||||
@ -22,7 +31,16 @@ pub fn dma2d_start(buffer: &[u8], pixels: i16) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn dma2d_start_blend(overlay_buffer: &[u8], bg_buffer: &[u8], pixels: i16) {
|
/// Starts blending
|
||||||
|
///
|
||||||
|
/// # Safety
|
||||||
|
///
|
||||||
|
/// This function is unsafe because the caller has to guarantee that he:
|
||||||
|
/// 1) doesn't mutate the buffers until the transfer is finished, which is
|
||||||
|
/// guaranteed by calling `dma2d_wait_for_transfer`
|
||||||
|
/// 2) the buffer doesn't get dropped until the transfer is finished
|
||||||
|
/// 3) doesn't call this function while another transfer is running
|
||||||
|
pub unsafe fn dma2d_start_blend(overlay_buffer: &[u8], bg_buffer: &[u8], pixels: i16) {
|
||||||
unsafe {
|
unsafe {
|
||||||
ffi::dma2d_start_blend(
|
ffi::dma2d_start_blend(
|
||||||
overlay_buffer.as_ptr() as _,
|
overlay_buffer.as_ptr() as _,
|
||||||
|
@ -8,7 +8,7 @@ use crate::ui::{
|
|||||||
|
|
||||||
#[cfg(feature = "dma2d")]
|
#[cfg(feature = "dma2d")]
|
||||||
use crate::trezorhal::{
|
use crate::trezorhal::{
|
||||||
buffers::{get_buffer_16bpp, get_buffer_4bpp},
|
buffers,
|
||||||
dma2d::{dma2d_setup_4bpp_over_4bpp, dma2d_start_blend, dma2d_wait_for_transfer},
|
dma2d::{dma2d_setup_4bpp_over_4bpp, dma2d_start_blend, dma2d_wait_for_transfer},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -353,11 +353,11 @@ pub fn loader_rust(
|
|||||||
|
|
||||||
let n_start = Point::new(-start_vector.y, start_vector.x);
|
let n_start = Point::new(-start_vector.y, start_vector.x);
|
||||||
|
|
||||||
let b1 = unsafe { get_buffer_16bpp(0, false) };
|
let mut b1 = buffers::BufferLine16bpp::get();
|
||||||
let b2 = unsafe { get_buffer_16bpp(1, false) };
|
let mut b2 = buffers::BufferLine16bpp::get();
|
||||||
let ib1 = unsafe { get_buffer_4bpp(0, true) };
|
let mut ib1 = buffers::BufferLine4bpp::get_cleared();
|
||||||
let ib2 = unsafe { get_buffer_4bpp(1, true) };
|
let mut ib2 = buffers::BufferLine4bpp::get_cleared();
|
||||||
let empty_line = unsafe { get_buffer_4bpp(2, true) };
|
let mut empty_line = buffers::BufferLine4bpp::get_cleared();
|
||||||
|
|
||||||
dma2d_setup_4bpp_over_4bpp(fg_color.into(), bg_color.into(), icon_color.into());
|
dma2d_setup_4bpp_over_4bpp(fg_color.into(), bg_color.into(), icon_color.into());
|
||||||
|
|
||||||
@ -420,8 +420,10 @@ pub fn loader_rust(
|
|||||||
}
|
}
|
||||||
|
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
|
unsafe {
|
||||||
dma2d_start_blend(&icon_buffer.buffer, &loader_buffer.buffer, clamped.width());
|
dma2d_start_blend(&icon_buffer.buffer, &loader_buffer.buffer, clamped.width());
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
}
|
}
|
||||||
|
@ -9,7 +9,6 @@ use super::{
|
|||||||
};
|
};
|
||||||
#[cfg(feature = "dma2d")]
|
#[cfg(feature = "dma2d")]
|
||||||
use crate::trezorhal::{
|
use crate::trezorhal::{
|
||||||
buffers::{get_buffer_16bpp, get_buffer_4bpp},
|
|
||||||
dma2d::{
|
dma2d::{
|
||||||
dma2d_setup_4bpp_over_16bpp, dma2d_setup_4bpp_over_4bpp, dma2d_start_blend,
|
dma2d_setup_4bpp_over_16bpp, dma2d_setup_4bpp_over_4bpp, dma2d_start_blend,
|
||||||
dma2d_wait_for_transfer,
|
dma2d_wait_for_transfer,
|
||||||
@ -21,7 +20,7 @@ use crate::ui::geometry::TOP_LEFT;
|
|||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
time::Duration,
|
time::Duration,
|
||||||
trezorhal::{buffers::get_text_buffer, display, time, uzlib::UzlibContext},
|
trezorhal::{buffers, display, time, uzlib::UzlibContext},
|
||||||
ui::lerp::Lerp,
|
ui::lerp::Lerp,
|
||||||
};
|
};
|
||||||
use core::slice;
|
use core::slice;
|
||||||
@ -448,13 +447,13 @@ pub fn text_over_image(
|
|||||||
offset_text: Offset,
|
offset_text: Offset,
|
||||||
text_color: Color,
|
text_color: Color,
|
||||||
) {
|
) {
|
||||||
let text_buffer = unsafe { get_text_buffer(0, true) };
|
let mut text_buffer = buffers::BufferText::get();
|
||||||
let img1 = unsafe { get_buffer_16bpp(0, true) };
|
let mut img1 = buffers::BufferLine16bpp::get_cleared();
|
||||||
let img2 = unsafe { get_buffer_16bpp(1, true) };
|
let mut img2 = buffers::BufferLine16bpp::get_cleared();
|
||||||
let empty_img = unsafe { get_buffer_16bpp(2, true) };
|
let mut empty_img = buffers::BufferLine16bpp::get_cleared();
|
||||||
let t1 = unsafe { get_buffer_4bpp(0, true) };
|
let mut t1 = buffers::BufferLine4bpp::get_cleared();
|
||||||
let t2 = unsafe { get_buffer_4bpp(1, true) };
|
let mut t2 = buffers::BufferLine4bpp::get_cleared();
|
||||||
let empty_t = unsafe { get_buffer_4bpp(2, true) };
|
let mut empty_t = buffers::BufferLine4bpp::get_cleared();
|
||||||
|
|
||||||
let r_img;
|
let r_img;
|
||||||
let area;
|
let area;
|
||||||
@ -495,7 +494,7 @@ pub fn text_over_image(
|
|||||||
Point::new(text_right, text_bottom),
|
Point::new(text_right, text_bottom),
|
||||||
);
|
);
|
||||||
|
|
||||||
display::text_into_buffer(text, font.into(), text_buffer, 0);
|
display::text_into_buffer(text, font.into(), &mut text_buffer, 0);
|
||||||
|
|
||||||
set_window(clamped);
|
set_window(clamped);
|
||||||
|
|
||||||
@ -548,7 +547,7 @@ pub fn text_over_image(
|
|||||||
}
|
}
|
||||||
|
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
dma2d_start_blend(&t_buffer.buffer, &img_buffer.buffer, clamped.width());
|
unsafe { dma2d_start_blend(&t_buffer.buffer, &img_buffer.buffer, clamped.width()) };
|
||||||
}
|
}
|
||||||
|
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
@ -572,12 +571,12 @@ pub fn icon_over_icon(
|
|||||||
fg: (Icon, Offset, Color),
|
fg: (Icon, Offset, Color),
|
||||||
bg_color: Color,
|
bg_color: Color,
|
||||||
) {
|
) {
|
||||||
let bg1 = unsafe { get_buffer_16bpp(0, true) };
|
let mut bg1 = buffers::BufferLine16bpp::get_cleared();
|
||||||
let bg2 = unsafe { get_buffer_16bpp(1, true) };
|
let mut bg2 = buffers::BufferLine16bpp::get_cleared();
|
||||||
let empty1 = unsafe { get_buffer_16bpp(2, true) };
|
let mut empty1 = buffers::BufferLine16bpp::get_cleared();
|
||||||
let fg1 = unsafe { get_buffer_4bpp(0, true) };
|
let mut fg1 = buffers::BufferLine4bpp::get_cleared();
|
||||||
let fg2 = unsafe { get_buffer_4bpp(1, true) };
|
let mut fg2 = buffers::BufferLine4bpp::get_cleared();
|
||||||
let empty2 = unsafe { get_buffer_4bpp(2, true) };
|
let mut empty2 = buffers::BufferLine4bpp::get_cleared();
|
||||||
|
|
||||||
let (icon_bg, offset_bg, color_icon_bg) = bg;
|
let (icon_bg, offset_bg, color_icon_bg) = bg;
|
||||||
let (icon_fg, offset_fg, color_icon_fg) = fg;
|
let (icon_fg, offset_fg, color_icon_fg) = fg;
|
||||||
@ -662,7 +661,7 @@ pub fn icon_over_icon(
|
|||||||
}
|
}
|
||||||
|
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
dma2d_start_blend(&fg_buffer.buffer, &bg_buffer.buffer, clamped.width());
|
unsafe { dma2d_start_blend(&fg_buffer.buffer, &bg_buffer.buffer, clamped.width()) };
|
||||||
}
|
}
|
||||||
|
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
@ -768,13 +767,13 @@ pub fn bar_with_text_and_fill(
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn marquee(area: Rect, text: &str, offset: i16, font: Font, fg: Color, bg: Color) {
|
pub fn marquee(area: Rect, text: &str, offset: i16, font: Font, fg: Color, bg: Color) {
|
||||||
let buffer = unsafe { get_text_buffer(0, true) };
|
let mut buffer = buffers::BufferText::get_cleared();
|
||||||
|
|
||||||
let area = area.translate(get_offset());
|
let area = area.translate(get_offset());
|
||||||
let clamped = area.clamp(constant::screen());
|
let clamped = area.clamp(constant::screen());
|
||||||
set_window(clamped);
|
set_window(clamped);
|
||||||
|
|
||||||
display::text_into_buffer(text, font.into(), buffer, offset);
|
display::text_into_buffer(text, font.into(), &mut buffer, offset);
|
||||||
let tbl = get_color_table(fg, bg);
|
let tbl = get_color_table(fg, bg);
|
||||||
|
|
||||||
for y in 0..clamped.height() {
|
for y in 0..clamped.height() {
|
||||||
|
@ -35,7 +35,7 @@ Trezor modifications:
|
|||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
trezorhal::{
|
trezorhal::{
|
||||||
buffers::{get_jpeg_work_buffer, BufferJpeg},
|
buffers::{BufferJpeg, BufferJpegWork},
|
||||||
display::pixeldata,
|
display::pixeldata,
|
||||||
},
|
},
|
||||||
ui::{
|
ui::{
|
||||||
@ -1399,7 +1399,8 @@ impl<'i, 'p> JDEC<'i, 'p> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn jpeg(data: &[u8], pos: Point, scale: u8) {
|
pub fn jpeg(data: &[u8], pos: Point, scale: u8) {
|
||||||
let pool = unsafe { get_jpeg_work_buffer(0, true).buffer.as_mut_slice() };
|
let mut buffer = BufferJpegWork::get_cleared();
|
||||||
|
let pool = buffer.buffer.as_mut_slice();
|
||||||
let mut out = PixelDataOutput(pos);
|
let mut out = PixelDataOutput(pos);
|
||||||
let mut inp = BufferInput(data);
|
let mut inp = BufferInput(data);
|
||||||
if let Ok(mut jd) = JDEC::new(&mut inp, pool) {
|
if let Ok(mut jd) = JDEC::new(&mut inp, pool) {
|
||||||
@ -1409,9 +1410,10 @@ pub fn jpeg(data: &[u8], pos: Point, scale: u8) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn jpeg_info(data: &[u8]) -> Option<(Offset, i16)> {
|
pub fn jpeg_info(data: &[u8]) -> Option<(Offset, i16)> {
|
||||||
let pool = unsafe { get_jpeg_work_buffer(0, true).buffer.as_mut_slice() };
|
let mut buffer = BufferJpegWork::get_cleared();
|
||||||
|
let pool = buffer.buffer.as_mut_slice();
|
||||||
let mut inp = BufferInput(data);
|
let mut inp = BufferInput(data);
|
||||||
if let Ok(jd) = JDEC::new(&mut inp, pool) {
|
let result = if let Ok(jd) = JDEC::new(&mut inp, pool) {
|
||||||
let mcu_height = jd.mcu_height();
|
let mcu_height = jd.mcu_height();
|
||||||
if mcu_height > 16 {
|
if mcu_height > 16 {
|
||||||
return None;
|
return None;
|
||||||
@ -1419,13 +1421,15 @@ pub fn jpeg_info(data: &[u8]) -> Option<(Offset, i16)> {
|
|||||||
Some((Offset::new(jd.width(), jd.height()), mcu_height))
|
Some((Offset::new(jd.width(), jd.height()), mcu_height))
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
};
|
||||||
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn jpeg_test(data: &[u8]) -> bool {
|
pub fn jpeg_test(data: &[u8]) -> bool {
|
||||||
let pool = unsafe { get_jpeg_work_buffer(0, true).buffer.as_mut_slice() };
|
let mut buffer = BufferJpegWork::get_cleared();
|
||||||
|
let pool = buffer.buffer.as_mut_slice();
|
||||||
let mut inp = BufferInput(data);
|
let mut inp = BufferInput(data);
|
||||||
if let Ok(mut jd) = JDEC::new(&mut inp, pool) {
|
let result = if let Ok(mut jd) = JDEC::new(&mut inp, pool) {
|
||||||
if jd.mcu_height() > 16 {
|
if jd.mcu_height() > 16 {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -1438,7 +1442,8 @@ pub fn jpeg_test(data: &[u8]) -> bool {
|
|||||||
res.is_ok()
|
res.is_ok()
|
||||||
} else {
|
} else {
|
||||||
false
|
false
|
||||||
}
|
};
|
||||||
|
result
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait JpegInput {
|
pub trait JpegInput {
|
||||||
@ -1464,18 +1469,18 @@ pub trait JpegOutput {
|
|||||||
fn write(&mut self, jd: &JDEC, rect: Rect, pixels: &[u16]) -> bool;
|
fn write(&mut self, jd: &JDEC, rect: Rect, pixels: &[u16]) -> bool;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct BufferOutput<'o> {
|
pub struct BufferOutput {
|
||||||
buffer: &'o mut BufferJpeg,
|
buffer: BufferJpeg,
|
||||||
buffer_width: i16,
|
buffer_width: i16,
|
||||||
buffer_height: i16,
|
buffer_height: i16,
|
||||||
current_line: i16,
|
current_line: i16,
|
||||||
current_line_pix: i16,
|
current_line_pix: i16,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'o> BufferOutput<'o> {
|
impl BufferOutput {
|
||||||
pub fn new(buffer: &'o mut BufferJpeg, buffer_width: i16, buffer_height: i16) -> Self {
|
pub fn new(buffer_width: i16, buffer_height: i16) -> Self {
|
||||||
Self {
|
Self {
|
||||||
buffer,
|
buffer: BufferJpeg::get_cleared(),
|
||||||
buffer_width,
|
buffer_width,
|
||||||
buffer_height,
|
buffer_height,
|
||||||
current_line: 0,
|
current_line: 0,
|
||||||
@ -1484,11 +1489,11 @@ impl<'o> BufferOutput<'o> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn buffer(&mut self) -> &mut BufferJpeg {
|
pub fn buffer(&mut self) -> &mut BufferJpeg {
|
||||||
self.buffer
|
&mut self.buffer
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'o> JpegOutput for BufferOutput<'o> {
|
impl JpegOutput for BufferOutput {
|
||||||
fn write(&mut self, jd: &JDEC, rect: Rect, bitmap: &[u16]) -> bool {
|
fn write(&mut self, jd: &JDEC, rect: Rect, bitmap: &[u16]) -> bool {
|
||||||
let w = rect.width();
|
let w = rect.width();
|
||||||
let h = rect.height();
|
let h = rect.height();
|
||||||
|
@ -15,7 +15,7 @@ use crate::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
trezorhal::{display::ToifFormat, uzlib::UZLIB_WINDOW_SIZE},
|
trezorhal::{buffers::BufferJpegWork, display::ToifFormat, uzlib::UZLIB_WINDOW_SIZE},
|
||||||
ui::{
|
ui::{
|
||||||
constant::HEIGHT,
|
constant::HEIGHT,
|
||||||
display::{tjpgd::BufferInput, toif::Toif},
|
display::{tjpgd::BufferInput, toif::Toif},
|
||||||
@ -211,7 +211,8 @@ where
|
|||||||
if let Ok(data) = res {
|
if let Ok(data) = res {
|
||||||
if is_image_jpeg(data.as_ref()) {
|
if is_image_jpeg(data.as_ref()) {
|
||||||
let mut input = BufferInput(data.as_ref());
|
let mut input = BufferInput(data.as_ref());
|
||||||
let mut hs_img = HomescreenJpeg::new(&mut input);
|
let mut pool = BufferJpegWork::get_cleared();
|
||||||
|
let mut hs_img = HomescreenJpeg::new(&mut input, pool.buffer.as_mut_slice());
|
||||||
homescreen(
|
homescreen(
|
||||||
&mut hs_img,
|
&mut hs_img,
|
||||||
&[text],
|
&[text],
|
||||||
@ -236,7 +237,8 @@ where
|
|||||||
|
|
||||||
if show_default {
|
if show_default {
|
||||||
let mut input = BufferInput(IMAGE_HOMESCREEN);
|
let mut input = BufferInput(IMAGE_HOMESCREEN);
|
||||||
let mut hs_img = HomescreenJpeg::new(&mut input);
|
let mut pool = BufferJpegWork::get_cleared();
|
||||||
|
let mut hs_img = HomescreenJpeg::new(&mut input, pool.buffer.as_mut_slice());
|
||||||
homescreen(
|
homescreen(
|
||||||
&mut hs_img,
|
&mut hs_img,
|
||||||
&[text],
|
&[text],
|
||||||
@ -328,7 +330,8 @@ where
|
|||||||
if let Ok(data) = res {
|
if let Ok(data) = res {
|
||||||
if is_image_jpeg(data.as_ref()) {
|
if is_image_jpeg(data.as_ref()) {
|
||||||
let mut input = BufferInput(data.as_ref());
|
let mut input = BufferInput(data.as_ref());
|
||||||
let mut hs_img = HomescreenJpeg::new(&mut input);
|
let mut pool = BufferJpegWork::get_cleared();
|
||||||
|
let mut hs_img = HomescreenJpeg::new(&mut input, pool.buffer.as_mut_slice());
|
||||||
homescreen_blurred(&mut hs_img, &texts);
|
homescreen_blurred(&mut hs_img, &texts);
|
||||||
show_default = false;
|
show_default = false;
|
||||||
} else if is_image_toif(data.as_ref()) {
|
} else if is_image_toif(data.as_ref()) {
|
||||||
@ -343,7 +346,8 @@ where
|
|||||||
|
|
||||||
if show_default {
|
if show_default {
|
||||||
let mut input = BufferInput(IMAGE_HOMESCREEN);
|
let mut input = BufferInput(IMAGE_HOMESCREEN);
|
||||||
let mut hs_img = HomescreenJpeg::new(&mut input);
|
let mut pool = BufferJpegWork::get_cleared();
|
||||||
|
let mut hs_img = HomescreenJpeg::new(&mut input, pool.buffer.as_mut_slice());
|
||||||
homescreen_blurred(&mut hs_img, &texts);
|
homescreen_blurred(&mut hs_img, &texts);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,31 +1,23 @@
|
|||||||
#[cfg(feature = "dma2d")]
|
|
||||||
use crate::trezorhal::{
|
|
||||||
buffers::{get_buffer_16bpp, get_buffer_4bpp, get_text_buffer, BufferText, LineBuffer4Bpp},
|
|
||||||
dma2d::{dma2d_setup_4bpp_over_16bpp, dma2d_start_blend, dma2d_wait_for_transfer},
|
|
||||||
};
|
|
||||||
use crate::{
|
use crate::{
|
||||||
trezorhal::{
|
trezorhal::{
|
||||||
buffers::{get_blurring_buffer, get_jpeg_buffer, get_jpeg_work_buffer, BufferJpeg},
|
buffers::{BufferBlurring, BufferJpeg, BufferLine16bpp, BufferLine4bpp, BufferText},
|
||||||
display,
|
display,
|
||||||
display::bar_radius_buffer,
|
display::bar_radius_buffer,
|
||||||
|
dma2d::{dma2d_setup_4bpp_over_16bpp, dma2d_start_blend, dma2d_wait_for_transfer},
|
||||||
uzlib::UzlibContext,
|
uzlib::UzlibContext,
|
||||||
},
|
},
|
||||||
ui::{
|
ui::{
|
||||||
constant::screen,
|
|
||||||
display::{position_buffer, set_window, Color},
|
|
||||||
geometry::{Offset, Point, Rect},
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
use crate::ui::{
|
|
||||||
component::text::TextStyle,
|
component::text::TextStyle,
|
||||||
constant::{HEIGHT, WIDTH},
|
constant::{screen, HEIGHT, WIDTH},
|
||||||
display::{
|
display::{
|
||||||
|
position_buffer, set_window,
|
||||||
tjpgd::{BufferInput, BufferOutput, JDEC},
|
tjpgd::{BufferInput, BufferOutput, JDEC},
|
||||||
Icon,
|
Color, Icon,
|
||||||
},
|
},
|
||||||
|
geometry::{Offset, Point, Rect},
|
||||||
model_tt::theme,
|
model_tt::theme,
|
||||||
util::icon_text_center,
|
util::icon_text_center,
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Clone, Copy)]
|
#[derive(Clone, Copy)]
|
||||||
@ -91,18 +83,15 @@ pub trait HomescreenDecompressor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub struct HomescreenJpeg<'i> {
|
pub struct HomescreenJpeg<'i> {
|
||||||
pub output: BufferOutput<'i>,
|
pub output: BufferOutput,
|
||||||
pub jdec: Option<JDEC<'i, 'i>>,
|
pub jdec: Option<JDEC<'i, 'i>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'i> HomescreenJpeg<'i> {
|
impl<'i> HomescreenJpeg<'i> {
|
||||||
pub fn new(input: &'i mut BufferInput<'i>) -> Self {
|
pub fn new(input: &'i mut BufferInput<'i>, pool: &'i mut [u8]) -> Self {
|
||||||
Self {
|
Self {
|
||||||
output: BufferOutput::new(unsafe { get_jpeg_buffer(0, true) }, WIDTH, 16),
|
output: BufferOutput::new(WIDTH, 16),
|
||||||
jdec: JDEC::new(input, unsafe {
|
jdec: JDEC::new(input, pool).ok(),
|
||||||
get_jpeg_work_buffer(0, true).buffer.as_mut_slice()
|
|
||||||
})
|
|
||||||
.ok(),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -125,7 +114,7 @@ impl<'i> HomescreenDecompressor for HomescreenJpeg<'i> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub struct HomescreenToif<'i> {
|
pub struct HomescreenToif<'i> {
|
||||||
pub output: BufferOutput<'i>,
|
pub output: BufferOutput,
|
||||||
pub decomp_context: UzlibContext<'i>,
|
pub decomp_context: UzlibContext<'i>,
|
||||||
line: i16,
|
line: i16,
|
||||||
}
|
}
|
||||||
@ -133,7 +122,7 @@ pub struct HomescreenToif<'i> {
|
|||||||
impl<'i> HomescreenToif<'i> {
|
impl<'i> HomescreenToif<'i> {
|
||||||
pub fn new(context: UzlibContext<'i>) -> Self {
|
pub fn new(context: UzlibContext<'i>) -> Self {
|
||||||
Self {
|
Self {
|
||||||
output: BufferOutput::new(unsafe { get_jpeg_buffer(0, true) }, WIDTH, 16),
|
output: BufferOutput::new(WIDTH, 16),
|
||||||
decomp_context: context,
|
decomp_context: context,
|
||||||
line: 0,
|
line: 0,
|
||||||
}
|
}
|
||||||
@ -184,7 +173,7 @@ fn homescreen_get_fg_text(
|
|||||||
y_tmp: i16,
|
y_tmp: i16,
|
||||||
text_info: HomescreenTextInfo,
|
text_info: HomescreenTextInfo,
|
||||||
text_buffer: &BufferText,
|
text_buffer: &BufferText,
|
||||||
fg_buffer: &mut LineBuffer4Bpp,
|
fg_buffer: &mut BufferLine4bpp,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
if y_tmp >= text_info.text_area.y0 && y_tmp < text_info.text_area.y1 {
|
if y_tmp >= text_info.text_area.y0 && y_tmp < text_info.text_area.y1 {
|
||||||
let y_pos = y_tmp - text_info.text_area.y0;
|
let y_pos = y_tmp - text_info.text_area.y0;
|
||||||
@ -204,7 +193,7 @@ fn homescreen_get_fg_icon(
|
|||||||
y_tmp: i16,
|
y_tmp: i16,
|
||||||
text_info: HomescreenTextInfo,
|
text_info: HomescreenTextInfo,
|
||||||
icon_data: &[u8],
|
icon_data: &[u8],
|
||||||
fg_buffer: &mut LineBuffer4Bpp,
|
fg_buffer: &mut BufferLine4bpp,
|
||||||
) {
|
) {
|
||||||
if let Some(icon_area) = text_info.icon_area {
|
if let Some(icon_area) = text_info.icon_area {
|
||||||
let icon_size = icon_area.size();
|
let icon_size = icon_area.size();
|
||||||
@ -291,13 +280,13 @@ fn homescreen_dim_area(x: i16, y: i16) -> bool {
|
|||||||
fn homescreen_line_blurred(
|
fn homescreen_line_blurred(
|
||||||
icon_data: &[u8],
|
icon_data: &[u8],
|
||||||
text_buffer: &mut BufferText,
|
text_buffer: &mut BufferText,
|
||||||
|
fg_buffer: &mut BufferLine4bpp,
|
||||||
|
img_buffer: &mut BufferLine16bpp,
|
||||||
text_info: HomescreenTextInfo,
|
text_info: HomescreenTextInfo,
|
||||||
blurring: &BlurringContext,
|
blurring: &BlurringContext,
|
||||||
y: i16,
|
y: i16,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
let t_buffer = unsafe { get_buffer_4bpp((y & 0x1) as u16, true) };
|
fg_buffer.buffer.fill(0);
|
||||||
let mut img_buffer = unsafe { get_buffer_16bpp((y & 0x1) as u16, false) };
|
|
||||||
|
|
||||||
for x in 0..HOMESCREEN_IMAGE_WIDTH {
|
for x in 0..HOMESCREEN_IMAGE_WIDTH {
|
||||||
let c = if LOCKSCREEN_DIM_ALL {
|
let c = if LOCKSCREEN_DIM_ALL {
|
||||||
let x = x as usize;
|
let x = x as usize;
|
||||||
@ -327,28 +316,31 @@ fn homescreen_line_blurred(
|
|||||||
img_buffer.buffer[j] = (c & 0xFF) as u8;
|
img_buffer.buffer[j] = (c & 0xFF) as u8;
|
||||||
}
|
}
|
||||||
|
|
||||||
let done = homescreen_get_fg_text(y, text_info, text_buffer, t_buffer);
|
let done = homescreen_get_fg_text(y, text_info, text_buffer, fg_buffer);
|
||||||
homescreen_get_fg_icon(y, text_info, icon_data, t_buffer);
|
homescreen_get_fg_icon(y, text_info, icon_data, fg_buffer);
|
||||||
|
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
dma2d_setup_4bpp_over_16bpp(text_info.text_color.into());
|
dma2d_setup_4bpp_over_16bpp(text_info.text_color.into());
|
||||||
dma2d_start_blend(&t_buffer.buffer, &img_buffer.buffer, WIDTH);
|
unsafe {
|
||||||
|
dma2d_start_blend(&fg_buffer.buffer, &img_buffer.buffer, WIDTH);
|
||||||
|
}
|
||||||
|
|
||||||
done
|
done
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::too_many_arguments)]
|
||||||
fn homescreen_line(
|
fn homescreen_line(
|
||||||
icon_data: &[u8],
|
icon_data: &[u8],
|
||||||
text_buffer: &mut BufferText,
|
text_buffer: &mut BufferText,
|
||||||
text_info: HomescreenTextInfo,
|
text_info: HomescreenTextInfo,
|
||||||
data_buffer: &mut BufferJpeg,
|
data_buffer: &mut BufferJpeg,
|
||||||
|
fg_buffer: &mut BufferLine4bpp,
|
||||||
|
img_buffer: &mut BufferLine16bpp,
|
||||||
mcu_height: i16,
|
mcu_height: i16,
|
||||||
y: i16,
|
y: i16,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
let t_buffer = unsafe { get_buffer_4bpp((y & 0x1) as u16, true) };
|
|
||||||
let mut img_buffer = unsafe { get_buffer_16bpp((y & 0x1) as u16, false) };
|
|
||||||
|
|
||||||
let image_data = get_data(data_buffer, y, mcu_height);
|
let image_data = get_data(data_buffer, y, mcu_height);
|
||||||
|
fg_buffer.buffer.fill(0);
|
||||||
|
|
||||||
for x in 0..HOMESCREEN_IMAGE_WIDTH {
|
for x in 0..HOMESCREEN_IMAGE_WIDTH {
|
||||||
let d = image_data[x as usize];
|
let d = image_data[x as usize];
|
||||||
@ -373,12 +365,14 @@ fn homescreen_line(
|
|||||||
img_buffer.buffer[j] = (c & 0xFF) as u8;
|
img_buffer.buffer[j] = (c & 0xFF) as u8;
|
||||||
}
|
}
|
||||||
|
|
||||||
let done = homescreen_get_fg_text(y, text_info, text_buffer, t_buffer);
|
let done = homescreen_get_fg_text(y, text_info, text_buffer, fg_buffer);
|
||||||
homescreen_get_fg_icon(y, text_info, icon_data, t_buffer);
|
homescreen_get_fg_icon(y, text_info, icon_data, fg_buffer);
|
||||||
|
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
dma2d_setup_4bpp_over_16bpp(text_info.text_color.into());
|
dma2d_setup_4bpp_over_16bpp(text_info.text_color.into());
|
||||||
dma2d_start_blend(&t_buffer.buffer, &img_buffer.buffer, WIDTH);
|
unsafe {
|
||||||
|
dma2d_start_blend(&fg_buffer.buffer, &img_buffer.buffer, WIDTH);
|
||||||
|
}
|
||||||
|
|
||||||
done
|
done
|
||||||
}
|
}
|
||||||
@ -427,7 +421,7 @@ fn update_accs_sub(data: &[u16], idx: usize, acc_r: &mut u16, acc_g: &mut u16, a
|
|||||||
}
|
}
|
||||||
|
|
||||||
struct BlurringContext {
|
struct BlurringContext {
|
||||||
pub lines: &'static mut [[[u16; 240usize]; 3usize]],
|
mem: BufferBlurring,
|
||||||
pub totals: [[u16; HOMESCREEN_IMAGE_WIDTH as usize]; COLORS],
|
pub totals: [[u16; HOMESCREEN_IMAGE_WIDTH as usize]; COLORS],
|
||||||
line_num: i16,
|
line_num: i16,
|
||||||
add_idx: usize,
|
add_idx: usize,
|
||||||
@ -436,9 +430,8 @@ struct BlurringContext {
|
|||||||
|
|
||||||
impl BlurringContext {
|
impl BlurringContext {
|
||||||
pub fn new() -> Self {
|
pub fn new() -> Self {
|
||||||
let mem = unsafe { get_blurring_buffer(0, true) };
|
|
||||||
Self {
|
Self {
|
||||||
lines: &mut mem.buffer[0..DECOMP_LINES],
|
mem: BufferBlurring::get_cleared(),
|
||||||
totals: [[0; HOMESCREEN_IMAGE_WIDTH as usize]; COLORS],
|
totals: [[0; HOMESCREEN_IMAGE_WIDTH as usize]; COLORS],
|
||||||
line_num: 0,
|
line_num: 0,
|
||||||
add_idx: 0,
|
add_idx: 0,
|
||||||
@ -447,8 +440,9 @@ impl BlurringContext {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn clear(&mut self) {
|
fn clear(&mut self) {
|
||||||
|
let lines = &mut self.mem.buffer[0..DECOMP_LINES];
|
||||||
for (i, total) in self.totals.iter_mut().enumerate() {
|
for (i, total) in self.totals.iter_mut().enumerate() {
|
||||||
for line in self.lines.iter_mut() {
|
for line in lines.iter_mut() {
|
||||||
line[i].fill(0);
|
line[i].fill(0);
|
||||||
}
|
}
|
||||||
total.fill(0);
|
total.fill(0);
|
||||||
@ -457,6 +451,7 @@ impl BlurringContext {
|
|||||||
|
|
||||||
// computes color averages for one line of image data
|
// computes color averages for one line of image data
|
||||||
fn compute_line_avgs(&mut self, buffer: &mut BufferJpeg, mcu_height: i16) {
|
fn compute_line_avgs(&mut self, buffer: &mut BufferJpeg, mcu_height: i16) {
|
||||||
|
let lines = &mut self.mem.buffer[0..DECOMP_LINES];
|
||||||
let mut acc_r = 0;
|
let mut acc_r = 0;
|
||||||
let mut acc_g = 0;
|
let mut acc_g = 0;
|
||||||
let mut acc_b = 0;
|
let mut acc_b = 0;
|
||||||
@ -468,9 +463,9 @@ impl BlurringContext {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for i in 0..HOMESCREEN_IMAGE_WIDTH {
|
for i in 0..HOMESCREEN_IMAGE_WIDTH {
|
||||||
self.lines[self.add_idx][RED_IDX][i as usize] = acc_r;
|
lines[self.add_idx][RED_IDX][i as usize] = acc_r;
|
||||||
self.lines[self.add_idx][GREEN_IDX][i as usize] = acc_g;
|
lines[self.add_idx][GREEN_IDX][i as usize] = acc_g;
|
||||||
self.lines[self.add_idx][BLUE_IDX][i as usize] = acc_b;
|
lines[self.add_idx][BLUE_IDX][i as usize] = acc_b;
|
||||||
|
|
||||||
// clamping handles left and right edges
|
// clamping handles left and right edges
|
||||||
let ic = (i - BLUR_RADIUS).clamp(0, HOMESCREEN_IMAGE_WIDTH - 1) as usize;
|
let ic = (i - BLUR_RADIUS).clamp(0, HOMESCREEN_IMAGE_WIDTH - 1) as usize;
|
||||||
@ -484,22 +479,24 @@ impl BlurringContext {
|
|||||||
|
|
||||||
// adds one line of averages to sliding total averages
|
// adds one line of averages to sliding total averages
|
||||||
fn vertical_avg_add(&mut self) {
|
fn vertical_avg_add(&mut self) {
|
||||||
|
let lines = &mut self.mem.buffer[0..DECOMP_LINES];
|
||||||
for i in 0..HOMESCREEN_IMAGE_WIDTH as usize {
|
for i in 0..HOMESCREEN_IMAGE_WIDTH as usize {
|
||||||
self.totals[RED_IDX][i] += self.lines[self.add_idx][RED_IDX][i];
|
self.totals[RED_IDX][i] += lines[self.add_idx][RED_IDX][i];
|
||||||
self.totals[GREEN_IDX][i] += self.lines[self.add_idx][GREEN_IDX][i];
|
self.totals[GREEN_IDX][i] += lines[self.add_idx][GREEN_IDX][i];
|
||||||
self.totals[BLUE_IDX][i] += self.lines[self.add_idx][BLUE_IDX][i];
|
self.totals[BLUE_IDX][i] += lines[self.add_idx][BLUE_IDX][i];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// adds one line and removes one line of averages to/from sliding total averages
|
// adds one line and removes one line of averages to/from sliding total averages
|
||||||
fn vertical_avg(&mut self) {
|
fn vertical_avg(&mut self) {
|
||||||
|
let lines = &mut self.mem.buffer[0..DECOMP_LINES];
|
||||||
for i in 0..HOMESCREEN_IMAGE_WIDTH as usize {
|
for i in 0..HOMESCREEN_IMAGE_WIDTH as usize {
|
||||||
self.totals[RED_IDX][i] +=
|
self.totals[RED_IDX][i] +=
|
||||||
self.lines[self.add_idx][RED_IDX][i] - self.lines[self.rem_idx][RED_IDX][i];
|
lines[self.add_idx][RED_IDX][i] - lines[self.rem_idx][RED_IDX][i];
|
||||||
self.totals[GREEN_IDX][i] +=
|
self.totals[GREEN_IDX][i] +=
|
||||||
self.lines[self.add_idx][GREEN_IDX][i] - self.lines[self.rem_idx][GREEN_IDX][i];
|
lines[self.add_idx][GREEN_IDX][i] - lines[self.rem_idx][GREEN_IDX][i];
|
||||||
self.totals[BLUE_IDX][i] +=
|
self.totals[BLUE_IDX][i] +=
|
||||||
self.lines[self.add_idx][BLUE_IDX][i] - self.lines[self.rem_idx][BLUE_IDX][i];
|
lines[self.add_idx][BLUE_IDX][i] - lines[self.rem_idx][BLUE_IDX][i];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -532,11 +529,15 @@ fn get_data(buffer: &mut BufferJpeg, line_num: i16, mcu_height: i16) -> &mut [u1
|
|||||||
pub fn homescreen_blurred(data: &mut dyn HomescreenDecompressor, texts: &[HomescreenText]) {
|
pub fn homescreen_blurred(data: &mut dyn HomescreenDecompressor, texts: &[HomescreenText]) {
|
||||||
let mut icon_data = [0_u8; (HOMESCREEN_MAX_ICON_SIZE * HOMESCREEN_MAX_ICON_SIZE / 2) as usize];
|
let mut icon_data = [0_u8; (HOMESCREEN_MAX_ICON_SIZE * HOMESCREEN_MAX_ICON_SIZE / 2) as usize];
|
||||||
|
|
||||||
let text_buffer = unsafe { get_text_buffer(0, true) };
|
let mut text_buffer = BufferText::get_cleared();
|
||||||
|
let mut fg_buffer_0 = BufferLine4bpp::get_cleared();
|
||||||
|
let mut img_buffer_0 = BufferLine16bpp::get_cleared();
|
||||||
|
let mut fg_buffer_1 = BufferLine4bpp::get_cleared();
|
||||||
|
let mut img_buffer_1 = BufferLine16bpp::get_cleared();
|
||||||
|
|
||||||
let mut next_text_idx = 1;
|
let mut next_text_idx = 1;
|
||||||
let mut text_info =
|
let mut text_info =
|
||||||
homescreen_position_text(unwrap!(texts.get(0)), text_buffer, &mut icon_data);
|
homescreen_position_text(unwrap!(texts.get(0)), &mut text_buffer, &mut icon_data);
|
||||||
|
|
||||||
let mcu_height = data.get_height();
|
let mcu_height = data.get_height();
|
||||||
data.decompress();
|
data.decompress();
|
||||||
@ -571,11 +572,36 @@ pub fn homescreen_blurred(data: &mut dyn HomescreenDecompressor, texts: &[Homesc
|
|||||||
blurring.compute_line_avgs(data.get_data(), mcu_height);
|
blurring.compute_line_avgs(data.get_data(), mcu_height);
|
||||||
}
|
}
|
||||||
|
|
||||||
let done = homescreen_line_blurred(&icon_data, text_buffer, text_info, &blurring, y);
|
let done = if y % 2 == 0 {
|
||||||
|
homescreen_line_blurred(
|
||||||
|
&icon_data,
|
||||||
|
&mut text_buffer,
|
||||||
|
&mut fg_buffer_0,
|
||||||
|
&mut img_buffer_0,
|
||||||
|
text_info,
|
||||||
|
&blurring,
|
||||||
|
y,
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
homescreen_line_blurred(
|
||||||
|
&icon_data,
|
||||||
|
&mut text_buffer,
|
||||||
|
&mut fg_buffer_1,
|
||||||
|
&mut img_buffer_1,
|
||||||
|
text_info,
|
||||||
|
&blurring,
|
||||||
|
y,
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
if done {
|
if done {
|
||||||
(text_info, next_text_idx) =
|
(text_info, next_text_idx) = homescreen_next_text(
|
||||||
homescreen_next_text(texts, text_buffer, &mut icon_data, text_info, next_text_idx);
|
texts,
|
||||||
|
&mut text_buffer,
|
||||||
|
&mut icon_data,
|
||||||
|
text_info,
|
||||||
|
next_text_idx,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
blurring.vertical_avg();
|
blurring.vertical_avg();
|
||||||
@ -613,7 +639,11 @@ pub fn homescreen(
|
|||||||
) {
|
) {
|
||||||
let mut icon_data = [0_u8; (HOMESCREEN_MAX_ICON_SIZE * HOMESCREEN_MAX_ICON_SIZE / 2) as usize];
|
let mut icon_data = [0_u8; (HOMESCREEN_MAX_ICON_SIZE * HOMESCREEN_MAX_ICON_SIZE / 2) as usize];
|
||||||
|
|
||||||
let text_buffer = unsafe { get_text_buffer(0, true) };
|
let mut text_buffer = BufferText::get_cleared();
|
||||||
|
let mut fg_buffer_0 = BufferLine4bpp::get_cleared();
|
||||||
|
let mut img_buffer_0 = BufferLine16bpp::get_cleared();
|
||||||
|
let mut fg_buffer_1 = BufferLine4bpp::get_cleared();
|
||||||
|
let mut img_buffer_1 = BufferLine16bpp::get_cleared();
|
||||||
|
|
||||||
let mut next_text_idx = 0;
|
let mut next_text_idx = 0;
|
||||||
let mut text_info = if let Some(notification) = notification {
|
let mut text_info = if let Some(notification) = notification {
|
||||||
@ -623,7 +653,7 @@ pub fn homescreen(
|
|||||||
WIDTH - NOTIFICATION_BORDER * 2,
|
WIDTH - NOTIFICATION_BORDER * 2,
|
||||||
NOTIFICATION_HEIGHT,
|
NOTIFICATION_HEIGHT,
|
||||||
2,
|
2,
|
||||||
text_buffer,
|
&mut text_buffer,
|
||||||
);
|
);
|
||||||
let area = Rect::new(
|
let area = Rect::new(
|
||||||
Point::new(0, NOTIFICATION_BORDER),
|
Point::new(0, NOTIFICATION_BORDER),
|
||||||
@ -637,7 +667,7 @@ pub fn homescreen(
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
next_text_idx += 1;
|
next_text_idx += 1;
|
||||||
homescreen_position_text(unwrap!(texts.get(0)), text_buffer, &mut icon_data)
|
homescreen_position_text(unwrap!(texts.get(0)), &mut text_buffer, &mut icon_data)
|
||||||
};
|
};
|
||||||
|
|
||||||
set_window(screen());
|
set_window(screen());
|
||||||
@ -649,14 +679,29 @@ pub fn homescreen(
|
|||||||
data.decompress();
|
data.decompress();
|
||||||
}
|
}
|
||||||
|
|
||||||
let done = homescreen_line(
|
let done = if y % 2 == 0 {
|
||||||
|
homescreen_line(
|
||||||
&icon_data,
|
&icon_data,
|
||||||
text_buffer,
|
&mut text_buffer,
|
||||||
text_info,
|
text_info,
|
||||||
data.get_data(),
|
data.get_data(),
|
||||||
|
&mut fg_buffer_0,
|
||||||
|
&mut img_buffer_0,
|
||||||
mcu_height,
|
mcu_height,
|
||||||
y,
|
y,
|
||||||
);
|
)
|
||||||
|
} else {
|
||||||
|
homescreen_line(
|
||||||
|
&icon_data,
|
||||||
|
&mut text_buffer,
|
||||||
|
text_info,
|
||||||
|
data.get_data(),
|
||||||
|
&mut fg_buffer_1,
|
||||||
|
&mut img_buffer_1,
|
||||||
|
mcu_height,
|
||||||
|
y,
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
if done {
|
if done {
|
||||||
if notification.is_some() && next_text_idx == 0 {
|
if notification.is_some() && next_text_idx == 0 {
|
||||||
@ -670,6 +715,9 @@ pub fn homescreen(
|
|||||||
|
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
|
|
||||||
|
drop(fg_buffer_0);
|
||||||
|
drop(fg_buffer_1);
|
||||||
|
|
||||||
icon_text_center(
|
icon_text_center(
|
||||||
text_info.text_area.center(),
|
text_info.text_area.center(),
|
||||||
notification.icon,
|
notification.icon,
|
||||||
@ -678,6 +726,10 @@ pub fn homescreen(
|
|||||||
style,
|
style,
|
||||||
Offset::new(1, -2),
|
Offset::new(1, -2),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
fg_buffer_0 = BufferLine4bpp::get_cleared();
|
||||||
|
fg_buffer_1 = BufferLine4bpp::get_cleared();
|
||||||
|
|
||||||
set_window(
|
set_window(
|
||||||
screen()
|
screen()
|
||||||
.split_top(NOTIFICATION_HEIGHT + NOTIFICATION_BORDER)
|
.split_top(NOTIFICATION_HEIGHT + NOTIFICATION_BORDER)
|
||||||
@ -690,8 +742,13 @@ pub fn homescreen(
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
(text_info, next_text_idx) =
|
(text_info, next_text_idx) = homescreen_next_text(
|
||||||
homescreen_next_text(texts, text_buffer, &mut icon_data, text_info, next_text_idx);
|
texts,
|
||||||
|
&mut text_buffer,
|
||||||
|
&mut icon_data,
|
||||||
|
text_info,
|
||||||
|
next_text_idx,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
dma2d_wait_for_transfer();
|
dma2d_wait_for_transfer();
|
||||||
|
Loading…
Reference in New Issue
Block a user