You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
256 lines
6.5 KiB
256 lines
6.5 KiB
/* |
|
* EEZ Modular Firmware |
|
* Copyright (C) 2015-present, Envox d.o.o. |
|
* |
|
* 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 3 of the License, or |
|
* (at your option) any later version. |
|
|
|
* This program is distributed in the hope that 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/>. |
|
*/ |
|
|
|
#include <assert.h> |
|
#include <stdlib.h> |
|
#include <memory.h> |
|
|
|
#include <eez/core/alloc.h> |
|
#include <eez/core/os.h> |
|
#include <eez/core/memory.h> |
|
#include <eez/core/debug.h> |
|
#include <eez/flow/flow.h> |
|
|
|
#include <eez/libs/lz4/lz4.h> |
|
|
|
#include <eez/gui/gui.h> |
|
#include <eez/gui/widget.h> |
|
|
|
#if OPTION_SCPI |
|
#include <scpi/scpi.h> |
|
#else |
|
#define SCPI_ERROR_OUT_OF_DEVICE_MEMORY -321 |
|
#define SCPI_ERROR_INVALID_BLOCK_DATA -161 |
|
#endif |
|
|
|
namespace eez { |
|
namespace gui { |
|
|
|
bool g_isMainAssetsLoaded; |
|
Assets *g_mainAssets; |
|
Assets *g_externalAssets; |
|
|
|
//////////////////////////////////////////////////////////////////////////////// |
|
|
|
void fixOffsets(Assets *assets); |
|
|
|
bool decompressAssetsData(const uint8_t *assetsData, uint32_t assetsDataSize, Assets *decompressedAssets, uint32_t maxDecompressedAssetsSize, int *err) { |
|
uint32_t compressedDataOffset; |
|
uint32_t decompressedSize; |
|
|
|
auto header = (Header *)assetsData; |
|
|
|
if (header->tag == HEADER_TAG) { |
|
decompressedAssets->projectMajorVersion = header->projectMajorVersion; |
|
decompressedAssets->projectMinorVersion = header->projectMinorVersion; |
|
decompressedAssets->assetsType = header->assetsType; |
|
|
|
compressedDataOffset = sizeof(Header); |
|
decompressedSize = header->decompressedSize; |
|
} else { |
|
decompressedAssets->projectMajorVersion = PROJECT_VERSION_V2; |
|
decompressedAssets->projectMinorVersion = 0; |
|
decompressedAssets->assetsType = ASSETS_TYPE_RESOURCE; |
|
|
|
compressedDataOffset = 4; |
|
decompressedSize = header->tag; |
|
} |
|
|
|
// disable warning: offsetof within non-standard-layout type ... is conditionally-supported [-Winvalid-offsetof] |
|
#ifdef __GNUC__ |
|
#pragma GCC diagnostic push |
|
#pragma GCC diagnostic ignored "-Winvalid-offsetof" |
|
#endif |
|
|
|
auto decompressedDataOffset = offsetof(Assets, settings); |
|
|
|
#ifdef __GNUC__ |
|
#pragma GCC diagnostic pop |
|
#endif |
|
|
|
|
|
if (decompressedDataOffset + decompressedSize > maxDecompressedAssetsSize) { |
|
if (err) { |
|
*err = SCPI_ERROR_OUT_OF_DEVICE_MEMORY; |
|
} |
|
return false; |
|
} |
|
|
|
int compressedSize = assetsDataSize - compressedDataOffset; |
|
|
|
int decompressResult = LZ4_decompress_safe( |
|
(const char *)(assetsData + compressedDataOffset), |
|
(char *)decompressedAssets + decompressedDataOffset, |
|
compressedSize, |
|
decompressedSize |
|
); |
|
|
|
if (decompressResult != (int)decompressedSize) { |
|
if (err) { |
|
*err = SCPI_ERROR_INVALID_BLOCK_DATA; |
|
} |
|
return false; |
|
} |
|
|
|
if (decompressedAssets->projectMajorVersion >= PROJECT_VERSION_V3) { |
|
fixOffsets(decompressedAssets); |
|
} |
|
|
|
return true; |
|
} |
|
|
|
void loadMainAssets(const uint8_t *assets, uint32_t assetsSize) { |
|
g_mainAssets = (Assets *)DECOMPRESSED_ASSETS_START_ADDRESS; |
|
g_mainAssets->external = false; |
|
auto decompressedSize = decompressAssetsData(assets, assetsSize, g_mainAssets, MAX_DECOMPRESSED_ASSETS_SIZE, nullptr); |
|
assert(decompressedSize); |
|
g_isMainAssetsLoaded = true; |
|
} |
|
|
|
void unloadExternalAssets() { |
|
if (g_externalAssets) { |
|
removeExternalPagesFromTheStack(); |
|
|
|
free(g_externalAssets); |
|
g_externalAssets = nullptr; |
|
} |
|
} |
|
|
|
//////////////////////////////////////////////////////////////////////////////// |
|
|
|
const PageAsset* getPageAsset(int pageId) { |
|
if (pageId > 0) { |
|
return g_mainAssets->pages[pageId - 1]; |
|
} else if (pageId < 0) { |
|
if (g_externalAssets == nullptr) { |
|
return nullptr; |
|
} |
|
return g_externalAssets->pages[-pageId - 1]; |
|
} |
|
return nullptr; |
|
} |
|
|
|
const PageAsset* getPageAsset(int pageId, WidgetCursor& widgetCursor) { |
|
if (pageId < 0) { |
|
widgetCursor.assets = g_externalAssets; |
|
widgetCursor.flowState = flow::getFlowState(g_externalAssets, -pageId - 1, widgetCursor); |
|
} else { |
|
widgetCursor.assets = g_mainAssets; |
|
if (g_mainAssets->flowDefinition) { |
|
widgetCursor.flowState = flow::getFlowState(g_mainAssets, pageId - 1, widgetCursor); |
|
} |
|
} |
|
return getPageAsset(pageId); |
|
} |
|
|
|
const Style *getStyle(int styleID) { |
|
if (styleID > 0) { |
|
return g_mainAssets->styles[styleID - 1]; |
|
} else if (styleID < 0) { |
|
if (g_externalAssets == nullptr) { |
|
return getStyle(STYLE_ID_DEFAULT); |
|
} |
|
return g_externalAssets->styles[-styleID - 1]; |
|
} |
|
return getStyle(STYLE_ID_DEFAULT); |
|
} |
|
|
|
const FontData *getFontData(int fontID) { |
|
if (fontID > 0) { |
|
return g_mainAssets->fonts[fontID - 1]; |
|
} else if (fontID < 0) { |
|
if (g_externalAssets == nullptr) { |
|
return nullptr; |
|
} |
|
return g_externalAssets->fonts[-fontID - 1]; |
|
} |
|
return nullptr; |
|
} |
|
|
|
const Bitmap *getBitmap(int bitmapID) { |
|
if (bitmapID > 0) { |
|
return g_mainAssets->bitmaps[bitmapID - 1]; |
|
} else if (bitmapID < 0) { |
|
if (g_externalAssets == nullptr) { |
|
return nullptr; |
|
} |
|
return g_externalAssets->bitmaps[-bitmapID - 1]; |
|
} |
|
return nullptr; |
|
} |
|
|
|
int getThemesCount() { |
|
return (int)g_mainAssets->colorsDefinition->themes.count; |
|
} |
|
|
|
Theme *getTheme(int i) { |
|
return g_mainAssets->colorsDefinition->themes[i]; |
|
} |
|
|
|
const char *getThemeName(int i) { |
|
return static_cast<const char *>(getTheme(i)->name); |
|
} |
|
|
|
const uint32_t getThemeColorsCount(int themeIndex) { |
|
return getTheme(themeIndex)->colors.count; |
|
} |
|
|
|
const uint16_t *getThemeColors(int themeIndex) { |
|
return static_cast<uint16_t *>(getTheme(themeIndex)->colors.items); |
|
} |
|
|
|
const uint16_t *getColors() { |
|
return static_cast<uint16_t *>(g_mainAssets->colorsDefinition->colors.items); |
|
} |
|
|
|
int getExternalAssetsMainPageId() { |
|
return -1; |
|
} |
|
|
|
const char *getActionName(const WidgetCursor &widgetCursor, int16_t actionId) { |
|
if (actionId == 0) { |
|
return nullptr; |
|
} |
|
|
|
if (actionId < 0) { |
|
actionId = -actionId; |
|
} |
|
actionId--; |
|
|
|
if (!widgetCursor.assets) { |
|
return ""; |
|
} |
|
|
|
return widgetCursor.assets->actionNames[actionId]; |
|
} |
|
|
|
int16_t getDataIdFromName(const WidgetCursor &widgetCursor, const char *name) { |
|
if (!widgetCursor.assets) { |
|
return 0; |
|
} |
|
|
|
for (uint32_t i = 0; i < widgetCursor.assets->variableNames.count; i++) { |
|
if (strcmp(widgetCursor.assets->variableNames[i], name) == 0) { |
|
return -((int16_t)i + 1); |
|
} |
|
} |
|
return 0; |
|
} |
|
|
|
} // namespace gui |
|
} // namespace eez
|
|
|