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.

257 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