mirror of
https://github.com/ByteWelder/Tactility.git
synced 2026-02-19 03:13:14 +00:00
- Refactor the way apps work: Instead of a C interface, they are now C++ classes. The main reasoning is that attaching data to an app was cumbersome. Having different implementations for different kinds of apps was cumbersome too. (3 or 4 layers of manifest nesting for the TactilityC project) - External apps are still written in C, but they get a createData/destroyData in their manifest, so: - External apps now have their own manifest. - All functions in the original AppManifest are removed and replaced by a single `createApp` function - External apps now automatically register (each app individually!) when they run the first time. As a side-effect they become visible in the `AppList` app! - Adapted all apps for the new interface. - Adapted all internal logic for these changes (Gui, ViewPort, Loader, AppContext, AppInstance, etc.) - Rewrote parts of Loader to use std::shared_ptr to make the code much safer. - Added a refcount check for the `AppInstance` and `App` at the end of their lifecycle. Show warning if refcount is too high.
84 lines
2.1 KiB
C++
84 lines
2.1 KiB
C++
#include "Mutex.h"
|
|
#include "Timer.h"
|
|
|
|
#include "service/ServiceContext.h"
|
|
#include "TactilityHeadless.h"
|
|
#include "service/ServiceRegistry.h"
|
|
|
|
#define TAG "sdcard_service"
|
|
|
|
namespace tt::service::sdcard {
|
|
|
|
extern const ServiceManifest manifest;
|
|
|
|
struct ServiceData {
|
|
Mutex mutex;
|
|
std::unique_ptr<Timer> updateTimer;
|
|
hal::SdCard::State lastState = hal::SdCard::State::Unmounted;
|
|
|
|
bool lock(TickType_t timeout) const {
|
|
return mutex.acquire(timeout) == TtStatusOk;
|
|
}
|
|
|
|
void unlock() const {
|
|
tt_check(mutex.release() == TtStatusOk);
|
|
}
|
|
};
|
|
|
|
static void onUpdate(std::shared_ptr<void> context) {
|
|
auto sdcard = tt::hal::getConfiguration()->sdcard;
|
|
if (sdcard == nullptr) {
|
|
return;
|
|
}
|
|
|
|
auto data = std::static_pointer_cast<ServiceData>(context);
|
|
|
|
if (!data->lock(50)) {
|
|
TT_LOG_W(TAG, LOG_MESSAGE_MUTEX_LOCK_FAILED);
|
|
return;
|
|
}
|
|
|
|
auto new_state = sdcard->getState();
|
|
|
|
if (new_state == hal::SdCard::State::Error) {
|
|
TT_LOG_W(TAG, "Sdcard error - unmounting. Did you eject the card in an unsafe manner?");
|
|
sdcard->unmount();
|
|
}
|
|
|
|
if (new_state != data->lastState) {
|
|
data->lastState = new_state;
|
|
}
|
|
|
|
data->unlock();
|
|
}
|
|
|
|
static void onStart(ServiceContext& service) {
|
|
if (hal::getConfiguration()->sdcard != nullptr) {
|
|
auto data = std::make_shared<ServiceData>();
|
|
service.setData(data);
|
|
|
|
data->updateTimer = std::make_unique<Timer>(Timer::Type::Periodic, onUpdate, data);
|
|
// We want to try and scan more often in case of startup or scan lock failure
|
|
data->updateTimer->start(1000);
|
|
} else {
|
|
TT_LOG_I(TAG, "task not started due to config");
|
|
}
|
|
}
|
|
|
|
static void onStop(ServiceContext& service) {
|
|
auto data = std::static_pointer_cast<ServiceData>(service.getData());
|
|
if (data->updateTimer != nullptr) {
|
|
// Stop thread
|
|
data->updateTimer->stop();
|
|
data->updateTimer = nullptr;
|
|
}
|
|
}
|
|
|
|
extern const ServiceManifest manifest = {
|
|
.id = "sdcard",
|
|
.onStart = onStart,
|
|
.onStop = onStop
|
|
};
|
|
|
|
} // namespace
|