- Update ILI9341 driver to v2.0.1 - Lots of code cleanup for apps - Refactor app "type" into "category" and added flags to the manifest (for show/hide statusbar and for hidden apps) - Rename some ElfApp-related functionality and improved the way the static data was managed - Rename "filebrowser" to "files" - Added cstring functions to tt_init.cpp - Minor fix in Boot app - Updated external apps for SDK changes
233 lines
6.7 KiB
C++
233 lines
6.7 KiB
C++
#ifdef ESP_PLATFORM
|
|
|
|
#include <Tactility/app/alertdialog/AlertDialog.h>
|
|
#include <Tactility/app/ElfApp.h>
|
|
#include <Tactility/file/File.h>
|
|
#include <Tactility/file/FileLock.h>
|
|
#include <Tactility/Log.h>
|
|
#include <Tactility/service/loader/Loader.h>
|
|
#include <Tactility/StringUtils.h>
|
|
|
|
#include <esp_elf.h>
|
|
|
|
#include <string>
|
|
#include <utility>
|
|
|
|
namespace tt::app {
|
|
|
|
constexpr auto* TAG = "ElfApp";
|
|
|
|
static std::string getErrorCodeString(int error_code) {
|
|
switch (error_code) {
|
|
case ENOMEM:
|
|
return "out of memory";
|
|
case ENOSYS:
|
|
return "missing symbol";
|
|
default:
|
|
return std::format("code {}", error_code);
|
|
}
|
|
}
|
|
|
|
class ElfApp final : public App {
|
|
|
|
public:
|
|
|
|
struct Parameters {
|
|
CreateData _Nullable createData = nullptr;
|
|
DestroyData _Nullable destroyData = nullptr;
|
|
OnCreate _Nullable onCreate = nullptr;
|
|
OnDestroy _Nullable onDestroy = nullptr;
|
|
OnShow _Nullable onShow = nullptr;
|
|
OnHide _Nullable onHide = nullptr;
|
|
OnResult _Nullable onResult = nullptr;
|
|
};
|
|
|
|
static void setParameters(const Parameters& parameters) {
|
|
staticParameters = parameters;
|
|
staticParametersSetCount++;
|
|
}
|
|
|
|
private:
|
|
|
|
static Parameters staticParameters;
|
|
static size_t staticParametersSetCount;
|
|
static std::shared_ptr<Lock> staticParametersLock;
|
|
|
|
const std::string appPath;
|
|
std::unique_ptr<uint8_t[]> elfFileData;
|
|
esp_elf_t elf {
|
|
.psegment = nullptr,
|
|
.svaddr = 0,
|
|
.ptext = nullptr,
|
|
.pdata = nullptr,
|
|
.sec = { },
|
|
.entry = nullptr
|
|
};
|
|
bool shouldCleanupElf = false; // Whether we have to clean up the above "elf" object
|
|
std::unique_ptr<Parameters> manifest;
|
|
void* data = nullptr;
|
|
std::string lastError = "";
|
|
|
|
bool startElf() {
|
|
const std::string elf_path = std::format("{}/elf/{}.elf", appPath, CONFIG_IDF_TARGET);
|
|
TT_LOG_I(TAG, "Starting ELF %s", elf_path.c_str());
|
|
assert(elfFileData == nullptr);
|
|
|
|
size_t size = 0;
|
|
file::withLock<void>(elf_path, [this, &elf_path, &size]{
|
|
elfFileData = file::readBinary(elf_path, size);
|
|
});
|
|
|
|
if (elfFileData == nullptr) {
|
|
return false;
|
|
}
|
|
|
|
if (esp_elf_init(&elf) != ESP_OK) {
|
|
lastError = "Failed to initialize";
|
|
TT_LOG_E(TAG, "%s", lastError.c_str());
|
|
elfFileData = nullptr;
|
|
return false;
|
|
}
|
|
|
|
auto relocate_result = esp_elf_relocate(&elf, elfFileData.get());
|
|
if (relocate_result != 0) {
|
|
// Note: the result code mapes to values from cstdlib's errno.h
|
|
lastError = getErrorCodeString(-relocate_result);
|
|
TT_LOG_E(TAG, "Application failed to load: %s", lastError.c_str());
|
|
elfFileData = nullptr;
|
|
return false;
|
|
}
|
|
|
|
int argc = 0;
|
|
char* argv[] = {};
|
|
|
|
if (esp_elf_request(&elf, 0, argc, argv) != ESP_OK) {
|
|
lastError = "Executable returned error code";
|
|
TT_LOG_E(TAG, "%s", lastError.c_str());
|
|
esp_elf_deinit(&elf);
|
|
elfFileData = nullptr;
|
|
return false;
|
|
}
|
|
|
|
shouldCleanupElf = true;
|
|
return true;
|
|
}
|
|
|
|
void stopElf() {
|
|
TT_LOG_I(TAG, "Cleaning up ELF");
|
|
|
|
if (shouldCleanupElf) {
|
|
esp_elf_deinit(&elf);
|
|
}
|
|
|
|
if (elfFileData != nullptr) {
|
|
elfFileData = nullptr;
|
|
}
|
|
}
|
|
|
|
public:
|
|
|
|
explicit ElfApp(std::string appPath) : appPath(std::move(appPath)) {}
|
|
|
|
void onCreate(AppContext& appContext) override {
|
|
// Because we use global variables, we have to ensure that we are not starting 2 apps in parallel
|
|
// We use a ScopedLock so we don't have to safeguard all branches
|
|
auto lock = staticParametersLock->asScopedLock();
|
|
lock.lock();
|
|
|
|
staticParametersSetCount = 0;
|
|
if (!startElf()) {
|
|
service::loader::stopApp();
|
|
auto message = lastError.empty() ? "Application failed to start." : std::format("Application failed to start: {}", lastError);
|
|
alertdialog::start("Error", message);
|
|
return;
|
|
}
|
|
|
|
if (staticParametersSetCount == 0) {
|
|
service::loader::stopApp();
|
|
alertdialog::start("Error", "Application failed to start: application failed to register itself");
|
|
return;
|
|
}
|
|
|
|
manifest = std::make_unique<Parameters>(staticParameters);
|
|
lock.unlock();
|
|
|
|
if (manifest->createData != nullptr) {
|
|
data = manifest->createData();
|
|
}
|
|
|
|
if (manifest->onCreate != nullptr) {
|
|
manifest->onCreate(&appContext, data);
|
|
}
|
|
}
|
|
|
|
void onDestroy(AppContext& appContext) override {
|
|
TT_LOG_I(TAG, "Cleaning up app");
|
|
if (manifest != nullptr) {
|
|
if (manifest->onDestroy != nullptr) {
|
|
manifest->onDestroy(&appContext, data);
|
|
}
|
|
|
|
if (manifest->destroyData != nullptr && data != nullptr) {
|
|
manifest->destroyData(data);
|
|
}
|
|
|
|
this->manifest = nullptr;
|
|
}
|
|
stopElf();
|
|
}
|
|
|
|
void onShow(AppContext& appContext, lv_obj_t* parent) override {
|
|
if (manifest != nullptr && manifest->onShow != nullptr) {
|
|
manifest->onShow(&appContext, data, parent);
|
|
}
|
|
}
|
|
|
|
void onHide(AppContext& appContext) override {
|
|
if (manifest != nullptr && manifest->onHide != nullptr) {
|
|
manifest->onHide(&appContext, data);
|
|
}
|
|
}
|
|
|
|
void onResult(AppContext& appContext, LaunchId launchId, Result result, std::unique_ptr<Bundle> resultBundle) override {
|
|
if (manifest != nullptr && manifest->onResult != nullptr) {
|
|
manifest->onResult(&appContext, data, launchId, result, resultBundle.get());
|
|
}
|
|
}
|
|
};
|
|
|
|
ElfApp::Parameters ElfApp::staticParameters;
|
|
size_t ElfApp::staticParametersSetCount = 0;
|
|
std::shared_ptr<Lock> ElfApp::staticParametersLock = std::make_shared<Mutex>();
|
|
|
|
void setElfAppParameters(
|
|
CreateData _Nullable createData,
|
|
DestroyData _Nullable destroyData,
|
|
OnCreate _Nullable onCreate,
|
|
OnDestroy _Nullable onDestroy,
|
|
OnShow _Nullable onShow,
|
|
OnHide _Nullable onHide,
|
|
OnResult _Nullable onResult
|
|
) {
|
|
ElfApp::setParameters({
|
|
.createData = createData,
|
|
.destroyData = destroyData,
|
|
.onCreate = onCreate,
|
|
.onDestroy = onDestroy,
|
|
.onShow = onShow,
|
|
.onHide = onHide,
|
|
.onResult = onResult
|
|
});
|
|
}
|
|
|
|
std::shared_ptr<App> createElfApp(const std::shared_ptr<AppManifest>& manifest) {
|
|
TT_LOG_I(TAG, "createElfApp");
|
|
assert(manifest != nullptr);
|
|
assert(manifest->location.isExternal());
|
|
return std::make_shared<ElfApp>(manifest->location.getPath());
|
|
}
|
|
|
|
} // namespace
|
|
|
|
#endif // ESP_PLATFORM
|