mirror of
https://github.com/ByteWelder/Tactility.git
synced 2026-02-18 19:03:16 +00:00
**New Features** * Runtime font accessors and new symbol fonts for text, launcher, statusbar, and shared icons. * Added font height base setting to device.properties * Text fonts now have 3 sizes: small, default, large **Improvements** * Renamed `UiScale` to `UiDensity` * Statusbar, toolbar and many UI components now compute heights and spacing from fonts/density. * SSD1306 initialization sequence refined for more stable startup. * Multiple image assets replaced by symbol-font rendering. * Many layout improvements related to density, font scaling and icon scaling * Updated folder name capitalization for newer style
374 lines
9.7 KiB
C++
374 lines
9.7 KiB
C++
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
#include <tactility/driver.h>
|
|
#include <tactility/device.h>
|
|
#include <tactility/error.h>
|
|
#include <tactility/log.h>
|
|
|
|
#include <ranges>
|
|
#include <cassert>
|
|
#include <cstring>
|
|
#include <sys/errno.h>
|
|
#include <vector>
|
|
|
|
#define TAG "device"
|
|
|
|
struct DeviceInternal {
|
|
/** Address of the API exposed by the device instance. */
|
|
struct Driver* driver = nullptr;
|
|
/** The driver data for this device (e.g. a mutex) */
|
|
void* driver_data = nullptr;
|
|
/** The mutex for device operations */
|
|
struct Mutex mutex {};
|
|
/** The device state */
|
|
struct {
|
|
int start_result = 0;
|
|
bool started : 1 = false;
|
|
bool added : 1 = false;
|
|
} state;
|
|
/** Attached child devices */
|
|
std::vector<Device*> children {};
|
|
};
|
|
|
|
struct DeviceLedger {
|
|
std::vector<Device*> devices;
|
|
Mutex mutex { 0 };
|
|
|
|
DeviceLedger() {
|
|
mutex_construct(&mutex);
|
|
}
|
|
|
|
~DeviceLedger() {
|
|
mutex_destruct(&mutex);
|
|
}
|
|
};
|
|
|
|
static DeviceLedger& get_ledger() {
|
|
static DeviceLedger ledger;
|
|
return ledger;
|
|
}
|
|
|
|
#define ledger get_ledger()
|
|
|
|
extern "C" {
|
|
|
|
#define ledger_lock() mutex_lock(&ledger.mutex)
|
|
#define ledger_unlock() mutex_unlock(&ledger.mutex)
|
|
|
|
#define lock_internal(internal) mutex_lock(&internal->mutex)
|
|
#define unlock_internal(internal) mutex_unlock(&internal->mutex)
|
|
|
|
error_t device_construct(Device* device) {
|
|
device->internal = new(std::nothrow) DeviceInternal;
|
|
if (device->internal == nullptr) {
|
|
return ERROR_OUT_OF_MEMORY;
|
|
}
|
|
LOG_D(TAG, "construct %s", device->name);
|
|
mutex_construct(&device->internal->mutex);
|
|
return ERROR_NONE;
|
|
}
|
|
|
|
error_t device_destruct(Device* device) {
|
|
lock_internal(device->internal);
|
|
|
|
auto* internal = device->internal;
|
|
|
|
if (internal->state.started || internal->state.added) {
|
|
unlock_internal(device->internal);
|
|
return ERROR_INVALID_STATE;
|
|
}
|
|
if (!internal->children.empty()) {
|
|
unlock_internal(device->internal);
|
|
return ERROR_INVALID_STATE;
|
|
}
|
|
LOG_D(TAG, "destruct %s", device->name);
|
|
|
|
device->internal = nullptr;
|
|
mutex_unlock(&internal->mutex);
|
|
delete internal;
|
|
|
|
return ERROR_NONE;
|
|
}
|
|
|
|
/** Add a child to the list of children */
|
|
static void device_add_child(struct Device* device, struct Device* child) {
|
|
device_lock(device);
|
|
check(device->internal->state.added);
|
|
device->internal->children.push_back(child);
|
|
device_unlock(device);
|
|
}
|
|
|
|
/** Remove a child from the list of children */
|
|
static void device_remove_child(struct Device* device, struct Device* child) {
|
|
device_lock(device);
|
|
const auto iterator = std::ranges::find(device->internal->children, child);
|
|
if (iterator != device->internal->children.end()) {
|
|
device->internal->children.erase(iterator);
|
|
}
|
|
device_unlock(device);
|
|
}
|
|
|
|
error_t device_add(Device* device) {
|
|
LOG_D(TAG, "add %s", device->name);
|
|
|
|
// Already added
|
|
if (device->internal->state.started || device->internal->state.added) {
|
|
return ERROR_INVALID_STATE;
|
|
}
|
|
|
|
// Add to ledger
|
|
ledger_lock();
|
|
ledger.devices.push_back(device);
|
|
ledger_unlock();
|
|
|
|
// Add self to parent's children list
|
|
auto* parent = device->parent;
|
|
if (parent != nullptr) {
|
|
device_add_child(parent, device);
|
|
}
|
|
|
|
device->internal->state.added = true;
|
|
return ERROR_NONE;
|
|
}
|
|
|
|
error_t device_remove(Device* device) {
|
|
LOG_D(TAG, "remove %s", device->name);
|
|
|
|
if (device->internal->state.started || !device->internal->state.added) {
|
|
return ERROR_INVALID_STATE;
|
|
}
|
|
|
|
// Remove self from parent's children list
|
|
auto* parent = device->parent;
|
|
if (parent != nullptr) {
|
|
device_remove_child(parent, device);
|
|
}
|
|
|
|
ledger_lock();
|
|
const auto iterator = std::ranges::find(ledger.devices, device);
|
|
if (iterator == ledger.devices.end()) {
|
|
ledger_unlock();
|
|
goto failed_ledger_lookup;
|
|
}
|
|
ledger.devices.erase(iterator);
|
|
ledger_unlock();
|
|
|
|
device->internal->state.added = false;
|
|
return ERROR_NONE;
|
|
|
|
failed_ledger_lookup:
|
|
|
|
// Re-add to parent
|
|
if (parent != nullptr) {
|
|
device_add_child(parent, device);
|
|
}
|
|
|
|
return ERROR_NOT_FOUND;
|
|
}
|
|
|
|
error_t device_start(Device* device) {
|
|
LOG_I(TAG, "start %s", device->name);
|
|
if (!device->internal->state.added) {
|
|
return ERROR_INVALID_STATE;
|
|
}
|
|
|
|
if (device->internal->driver == nullptr) {
|
|
return ERROR_INVALID_STATE;
|
|
}
|
|
|
|
// Already started
|
|
if (device->internal->state.started) {
|
|
return ERROR_NONE;
|
|
}
|
|
|
|
error_t bind_error = driver_bind(device->internal->driver, device);
|
|
device->internal->state.started = (bind_error == ERROR_NONE);
|
|
device->internal->state.start_result = bind_error;
|
|
return bind_error == ERROR_NONE ? ERROR_NONE : ERROR_RESOURCE;
|
|
}
|
|
|
|
error_t device_stop(struct Device* device) {
|
|
LOG_I(TAG, "stop %s", device->name);
|
|
if (!device->internal->state.added) {
|
|
return ERROR_INVALID_STATE;
|
|
}
|
|
|
|
if (!device->internal->state.started) {
|
|
return ERROR_NONE;
|
|
}
|
|
|
|
if (driver_unbind(device->internal->driver, device) != ERROR_NONE) {
|
|
return ERROR_RESOURCE;
|
|
}
|
|
|
|
device->internal->state.started = false;
|
|
device->internal->state.start_result = 0;
|
|
return ERROR_NONE;
|
|
}
|
|
|
|
error_t device_construct_add(struct Device* device, const char* compatible) {
|
|
struct Driver* driver = driver_find_compatible(compatible);
|
|
if (driver == nullptr) {
|
|
LOG_E(TAG, "Can't find driver '%s' for device '%s'", compatible, device->name);
|
|
return ERROR_RESOURCE;
|
|
}
|
|
|
|
error_t error = device_construct(device);
|
|
if (error != ERROR_NONE) {
|
|
LOG_E(TAG, "Failed to construct device %s: %s", device->name, error_to_string(error));
|
|
goto on_construct_error;
|
|
}
|
|
|
|
device_set_driver(device, driver);
|
|
|
|
error = device_add(device);
|
|
if (error != ERROR_NONE) {
|
|
LOG_E(TAG, "Failed to add device %s: %s", device->name, error_to_string(error));
|
|
goto on_add_error;
|
|
}
|
|
|
|
return ERROR_NONE;
|
|
|
|
on_add_error:
|
|
device_destruct(device);
|
|
on_construct_error:
|
|
return error;
|
|
}
|
|
|
|
error_t device_construct_add_start(struct Device* device, const char* compatible) {
|
|
error_t error = device_construct_add(device, compatible);
|
|
if (error != ERROR_NONE) {
|
|
goto on_construct_add_error;
|
|
}
|
|
|
|
error = device_start(device);
|
|
if (error != ERROR_NONE) {
|
|
LOG_E(TAG, "Failed to start device %s: %s", device->name, error_to_string(error));
|
|
goto on_start_error;
|
|
}
|
|
|
|
return ERROR_NONE;
|
|
|
|
on_start_error:
|
|
device_remove(device);
|
|
device_destruct(device);
|
|
on_construct_add_error:
|
|
return error;
|
|
}
|
|
|
|
void device_set_parent(Device* device, Device* parent) {
|
|
assert(!device->internal->state.started);
|
|
device->parent = parent;
|
|
}
|
|
|
|
Device* device_get_parent(struct Device* device) {
|
|
return device->parent;
|
|
}
|
|
|
|
void device_set_driver(struct Device* device, struct Driver* driver) {
|
|
device->internal->driver = driver;
|
|
}
|
|
|
|
struct Driver* device_get_driver(struct Device* device) {
|
|
return device->internal->driver;
|
|
}
|
|
|
|
bool device_is_ready(const struct Device* device) {
|
|
return device->internal->state.started;
|
|
}
|
|
|
|
bool device_is_compatible(const struct Device* device, const char* compatible) {
|
|
if (device->internal->driver == nullptr) return false;
|
|
return driver_is_compatible(device->internal->driver, compatible);
|
|
}
|
|
|
|
void device_set_driver_data(struct Device* device, void* driver_data) {
|
|
device->internal->driver_data = driver_data;
|
|
}
|
|
|
|
void* device_get_driver_data(struct Device* device) {
|
|
return device->internal->driver_data;
|
|
}
|
|
|
|
bool device_is_added(const struct Device* device) {
|
|
return device->internal->state.added;
|
|
}
|
|
|
|
void device_lock(struct Device* device) {
|
|
mutex_lock(&device->internal->mutex);
|
|
}
|
|
|
|
bool device_try_lock(struct Device* device, TickType_t timeout) {
|
|
return mutex_try_lock(&device->internal->mutex, timeout);
|
|
}
|
|
|
|
void device_unlock(struct Device* device) {
|
|
mutex_unlock(&device->internal->mutex);
|
|
}
|
|
|
|
const struct DeviceType* device_get_type(struct Device* device) {
|
|
return device->internal->driver ? device->internal->driver->device_type : NULL;
|
|
}
|
|
|
|
void device_for_each(void* callback_context, bool(*on_device)(Device* device, void* context)) {
|
|
ledger_lock();
|
|
for (auto* device : ledger.devices) {
|
|
if (!on_device(device, callback_context)) {
|
|
break;
|
|
}
|
|
}
|
|
ledger_unlock();
|
|
}
|
|
|
|
void device_for_each_child(Device* device, void* callbackContext, bool(*on_device)(struct Device* device, void* context)) {
|
|
for (auto* child_device : device->internal->children) {
|
|
if (!on_device(child_device, callbackContext)) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void device_for_each_of_type(const DeviceType* type, void* callbackContext, bool(*on_device)(Device* device, void* context)) {
|
|
ledger_lock();
|
|
for (auto* device : ledger.devices) {
|
|
auto* driver = device->internal->driver;
|
|
if (driver != nullptr) {
|
|
if (driver->device_type == type) {
|
|
if (!on_device(device, callbackContext)) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
ledger_unlock();
|
|
}
|
|
|
|
bool device_exists_of_type(const DeviceType* type) {
|
|
bool found = false;
|
|
ledger_lock();
|
|
for (auto* device : ledger.devices) {
|
|
auto* driver = device->internal->driver;
|
|
if (driver != nullptr && driver->device_type == type) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
ledger_unlock();
|
|
return found;
|
|
}
|
|
|
|
Device* device_find_by_name(const char* name) {
|
|
Device* found = nullptr;
|
|
ledger_lock();
|
|
for (auto* device : ledger.devices) {
|
|
if (device->name != nullptr && std::strcmp(device->name, name) == 0) {
|
|
found = device;
|
|
break;
|
|
}
|
|
}
|
|
ledger_unlock();
|
|
return found;
|
|
}
|
|
|
|
} // extern "C"
|