mirror of
https://github.com/UberGuidoZ/Flipper.git
synced 2025-01-05 13:20:22 +00:00
367 lines
13 KiB
C
367 lines
13 KiB
C
|
#include <furi.h>
|
||
|
#include <furi_hal_gpio.h>
|
||
|
#include <furi_hal_power.h>
|
||
|
#include <gui/gui.h>
|
||
|
#include "coleco_icons.h"
|
||
|
|
||
|
#define CODE_0 0x0A
|
||
|
#define CODE_1 0x0D
|
||
|
#define CODE_2 0x07
|
||
|
#define CODE_3 0x0C
|
||
|
#define CODE_4 0x02
|
||
|
#define CODE_5 0x03
|
||
|
#define CODE_6 0x0E
|
||
|
#define CODE_7 0x05
|
||
|
#define CODE_8 0x01
|
||
|
#define CODE_9 0x0B
|
||
|
#define CODE_H 0x06
|
||
|
#define CODE_S 0x09
|
||
|
#define CODE_N 0x0F
|
||
|
|
||
|
const GpioPin* const pin_up = &gpio_ext_pa6;
|
||
|
const GpioPin* const pin_down = &gpio_ext_pc0;
|
||
|
const GpioPin* const pin_right = &gpio_ext_pb2;
|
||
|
const GpioPin* const pin_left = &gpio_ext_pc3;
|
||
|
const GpioPin* const pin_code0 = &gpio_ext_pa7;
|
||
|
const GpioPin* const pin_code1 = &gpio_ext_pa4;
|
||
|
const GpioPin* const pin_code2 = &ibutton_gpio;
|
||
|
const GpioPin* const pin_code3 = &gpio_ext_pc1;
|
||
|
const GpioPin* const pin_fire = &gpio_ext_pb3;
|
||
|
const GpioPin* const pin_alt = &gpio_usart_tx;
|
||
|
|
||
|
typedef enum {
|
||
|
EventTypeTick,
|
||
|
EventTypeKey,
|
||
|
} EventType;
|
||
|
|
||
|
typedef struct {
|
||
|
EventType type;
|
||
|
InputEvent input;
|
||
|
} PluginEvent;
|
||
|
|
||
|
typedef struct {
|
||
|
bool dpad;
|
||
|
int row;
|
||
|
int column;
|
||
|
} Coleco;
|
||
|
|
||
|
static void render_callback(Canvas* const canvas, void* context) {
|
||
|
Coleco* coleco = acquire_mutex((ValueMutex*)context, 25);
|
||
|
if(coleco == NULL) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if(coleco->dpad) {
|
||
|
canvas_draw_icon(canvas, 4, 16, &I_ColecoJoystick_sel_33x33);
|
||
|
canvas_draw_icon(canvas, 27, 52, &I_ColecoFire_sel_18x9);
|
||
|
} else {
|
||
|
const bool hvr = coleco->row == 0 && coleco->column < 2;
|
||
|
canvas_draw_icon(
|
||
|
canvas, 4, 16, hvr ? &I_ColecoJoystick_hvr_33x33 : &I_ColecoJoystick_33x33);
|
||
|
canvas_draw_icon(canvas, 27, 52, hvr ? &I_ColecoFire_hvr_18x9 : &I_ColecoFire_18x9);
|
||
|
}
|
||
|
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
27,
|
||
|
4,
|
||
|
(coleco->row == 0 && coleco->column == 2) ? &I_ColecoAlt_hvr_18x9 : &I_ColecoAlt_18x9);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
49,
|
||
|
44,
|
||
|
(coleco->row == 1 && coleco->column == 0) ? &I_Coleco1_hvr_17x17 : &I_Coleco1_17x17);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
49,
|
||
|
24,
|
||
|
(coleco->row == 1 && coleco->column == 1) ? &I_Coleco2_hvr_17x17 : &I_Coleco2_17x17);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
49,
|
||
|
4,
|
||
|
(coleco->row == 1 && coleco->column == 2) ? &I_Coleco3_hvr_17x17 : &I_Coleco3_17x17);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
69,
|
||
|
44,
|
||
|
(coleco->row == 2 && coleco->column == 0) ? &I_Coleco4_hvr_17x17 : &I_Coleco4_17x17);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
69,
|
||
|
24,
|
||
|
(coleco->row == 2 && coleco->column == 1) ? &I_Coleco5_hvr_17x17 : &I_Coleco5_17x17);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
69,
|
||
|
4,
|
||
|
(coleco->row == 2 && coleco->column == 2) ? &I_Coleco6_hvr_17x17 : &I_Coleco6_17x17);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
89,
|
||
|
44,
|
||
|
(coleco->row == 3 && coleco->column == 0) ? &I_Coleco7_hvr_17x17 : &I_Coleco7_17x17);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
89,
|
||
|
24,
|
||
|
(coleco->row == 3 && coleco->column == 1) ? &I_Coleco8_hvr_17x17 : &I_Coleco8_17x17);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
89,
|
||
|
4,
|
||
|
(coleco->row == 3 && coleco->column == 2) ? &I_Coleco9_hvr_17x17 : &I_Coleco9_17x17);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
109,
|
||
|
44,
|
||
|
(coleco->row == 4 && coleco->column == 0) ? &I_ColecoStar_hvr_17x17 : &I_ColecoStar_17x17);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
109,
|
||
|
24,
|
||
|
(coleco->row == 4 && coleco->column == 1) ? &I_Coleco0_hvr_17x17 : &I_Coleco0_17x17);
|
||
|
canvas_draw_icon(
|
||
|
canvas,
|
||
|
109,
|
||
|
4,
|
||
|
(coleco->row == 4 && coleco->column == 2) ? &I_ColecoPound_hvr_17x17 :
|
||
|
&I_ColecoPound_17x17);
|
||
|
|
||
|
release_mutex((ValueMutex*)context, coleco);
|
||
|
}
|
||
|
|
||
|
static void input_callback(InputEvent* input_event, FuriMessageQueue* event_queue) {
|
||
|
furi_assert(event_queue);
|
||
|
|
||
|
PluginEvent event = {.type = EventTypeKey, .input = *input_event};
|
||
|
furi_message_queue_put(event_queue, &event, FuriWaitForever);
|
||
|
}
|
||
|
|
||
|
static void coleco_write_code(uint8_t code) {
|
||
|
furi_hal_gpio_write(pin_code0, (code & 1));
|
||
|
furi_hal_gpio_write(pin_code1, (code & 2));
|
||
|
furi_hal_gpio_write(pin_code2, (code & 4));
|
||
|
furi_hal_gpio_write(pin_code3, (code & 8));
|
||
|
}
|
||
|
|
||
|
static void coleco_gpio_init() {
|
||
|
// configure output pins
|
||
|
furi_hal_gpio_init(pin_up, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
|
||
|
furi_hal_gpio_init(pin_down, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
|
||
|
furi_hal_gpio_init(pin_right, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
|
||
|
furi_hal_gpio_init(pin_left, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
|
||
|
furi_hal_gpio_init(pin_code0, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
|
||
|
furi_hal_gpio_init(pin_code1, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
|
||
|
furi_hal_gpio_init(pin_code2, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
|
||
|
furi_hal_gpio_init(pin_code3, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
|
||
|
furi_hal_gpio_init(pin_fire, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
|
||
|
furi_hal_gpio_init(pin_alt, GpioModeOutputPushPull, GpioPullNo, GpioSpeedVeryHigh);
|
||
|
|
||
|
furi_hal_gpio_write(pin_up, true);
|
||
|
furi_hal_gpio_write(pin_down, true);
|
||
|
furi_hal_gpio_write(pin_right, true);
|
||
|
furi_hal_gpio_write(pin_left, true);
|
||
|
furi_hal_gpio_write(pin_fire, true);
|
||
|
furi_hal_gpio_write(pin_alt, true);
|
||
|
|
||
|
coleco_write_code(CODE_N);
|
||
|
}
|
||
|
|
||
|
static Coleco* coleco_alloc() {
|
||
|
Coleco* coleco = malloc(sizeof(Coleco));
|
||
|
|
||
|
coleco->dpad = false;
|
||
|
coleco->row = 0;
|
||
|
coleco->column = 1;
|
||
|
|
||
|
return coleco;
|
||
|
}
|
||
|
|
||
|
static void coleco_free(Coleco* coleco) {
|
||
|
furi_assert(coleco);
|
||
|
|
||
|
free(coleco);
|
||
|
}
|
||
|
|
||
|
int32_t coleco_app(void* p) {
|
||
|
UNUSED(p);
|
||
|
|
||
|
FuriMessageQueue* event_queue = furi_message_queue_alloc(8, sizeof(PluginEvent));
|
||
|
|
||
|
Coleco* coleco = coleco_alloc();
|
||
|
|
||
|
ValueMutex coleco_mutex;
|
||
|
if(!init_mutex(&coleco_mutex, coleco, sizeof(Coleco))) {
|
||
|
FURI_LOG_E("Coleco", "cannot create mutex\r\n");
|
||
|
coleco_free(coleco);
|
||
|
return 255;
|
||
|
}
|
||
|
|
||
|
// set system callbacks
|
||
|
ViewPort* view_port = view_port_alloc();
|
||
|
view_port_draw_callback_set(view_port, render_callback, &coleco_mutex);
|
||
|
view_port_input_callback_set(view_port, input_callback, event_queue);
|
||
|
|
||
|
// open GUI and register view_port
|
||
|
Gui* gui = furi_record_open("gui");
|
||
|
gui_add_view_port(gui, view_port, GuiLayerFullscreen);
|
||
|
|
||
|
coleco_gpio_init();
|
||
|
furi_hal_power_enable_otg();
|
||
|
|
||
|
PluginEvent event;
|
||
|
for(bool processing = true; processing;) {
|
||
|
FuriStatus event_status = furi_message_queue_get(event_queue, &event, 100);
|
||
|
|
||
|
Coleco* coleco = (Coleco*)acquire_mutex_block(&coleco_mutex);
|
||
|
|
||
|
if(event_status == FuriStatusOk) {
|
||
|
// press events
|
||
|
if(event.type == EventTypeKey) {
|
||
|
switch(event.input.key) {
|
||
|
case InputKeyUp:
|
||
|
if(coleco->dpad) {
|
||
|
if(event.input.type == InputTypePress) {
|
||
|
furi_hal_gpio_write(pin_up, false);
|
||
|
} else if(event.input.type == InputTypeRelease) {
|
||
|
furi_hal_gpio_write(pin_up, true);
|
||
|
}
|
||
|
} else {
|
||
|
if(event.input.type == InputTypePress && coleco->column < 2) {
|
||
|
coleco->column++;
|
||
|
coleco_write_code(CODE_N);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case InputKeyDown:
|
||
|
if(coleco->dpad) {
|
||
|
if(event.input.type == InputTypePress) {
|
||
|
furi_hal_gpio_write(pin_down, false);
|
||
|
} else if(event.input.type == InputTypeRelease) {
|
||
|
furi_hal_gpio_write(pin_down, true);
|
||
|
}
|
||
|
} else {
|
||
|
if(event.input.type == InputTypePress && coleco->column > 0) {
|
||
|
coleco->column--;
|
||
|
coleco_write_code(CODE_N);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case InputKeyRight:
|
||
|
if(coleco->dpad) {
|
||
|
if(event.input.type == InputTypePress) {
|
||
|
furi_hal_gpio_write(pin_right, false);
|
||
|
} else if(event.input.type == InputTypeRelease) {
|
||
|
furi_hal_gpio_write(pin_right, true);
|
||
|
}
|
||
|
} else {
|
||
|
if(event.input.type == InputTypePress && coleco->row < 4) {
|
||
|
coleco->row++;
|
||
|
coleco_write_code(CODE_N);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case InputKeyLeft:
|
||
|
if(coleco->dpad) {
|
||
|
if(event.input.type == InputTypePress) {
|
||
|
furi_hal_gpio_write(pin_left, false);
|
||
|
} else if(event.input.type == InputTypeRelease) {
|
||
|
furi_hal_gpio_write(pin_left, true);
|
||
|
}
|
||
|
} else {
|
||
|
if(event.input.type == InputTypePress && coleco->row > 0) {
|
||
|
coleco->row--;
|
||
|
coleco_write_code(CODE_N);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case InputKeyOk:
|
||
|
if(coleco->dpad) {
|
||
|
if(event.input.type == InputTypePress) {
|
||
|
furi_hal_gpio_write(pin_fire, false);
|
||
|
} else if(event.input.type == InputTypeRelease) {
|
||
|
furi_hal_gpio_write(pin_fire, true);
|
||
|
}
|
||
|
} else {
|
||
|
if(event.input.type == InputTypePress) {
|
||
|
if(coleco->row == 0) {
|
||
|
if(coleco->column == 2) {
|
||
|
furi_hal_gpio_write(pin_alt, false);
|
||
|
} else {
|
||
|
coleco->dpad = true;
|
||
|
}
|
||
|
} else if(coleco->row == 1) {
|
||
|
if(coleco->column == 0) {
|
||
|
coleco_write_code(CODE_1);
|
||
|
} else if(coleco->column == 1) {
|
||
|
coleco_write_code(CODE_2);
|
||
|
} else {
|
||
|
coleco_write_code(CODE_3);
|
||
|
}
|
||
|
} else if(coleco->row == 2) {
|
||
|
if(coleco->column == 0) {
|
||
|
coleco_write_code(CODE_4);
|
||
|
} else if(coleco->column == 1) {
|
||
|
coleco_write_code(CODE_5);
|
||
|
} else {
|
||
|
coleco_write_code(CODE_6);
|
||
|
}
|
||
|
} else if(coleco->row == 3) {
|
||
|
if(coleco->column == 0) {
|
||
|
coleco_write_code(CODE_7);
|
||
|
} else if(coleco->column == 1) {
|
||
|
coleco_write_code(CODE_8);
|
||
|
} else {
|
||
|
coleco_write_code(CODE_9);
|
||
|
}
|
||
|
} else if(coleco->row == 4) {
|
||
|
if(coleco->column == 0) {
|
||
|
coleco_write_code(CODE_S);
|
||
|
} else if(coleco->column == 1) {
|
||
|
coleco_write_code(CODE_0);
|
||
|
} else {
|
||
|
coleco_write_code(CODE_H);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if(event.input.type == InputTypeRelease) {
|
||
|
furi_hal_gpio_write(pin_alt, true);
|
||
|
coleco_write_code(CODE_N);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case InputKeyBack:
|
||
|
if(event.input.type == InputTypePress) {
|
||
|
if(coleco->dpad) {
|
||
|
coleco->dpad = false;
|
||
|
} else {
|
||
|
processing = false;
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
view_port_update(view_port);
|
||
|
}
|
||
|
} else {
|
||
|
FURI_LOG_D("Coleco", "FuriMessageQueue: event timeout");
|
||
|
}
|
||
|
|
||
|
release_mutex(&coleco_mutex, coleco);
|
||
|
}
|
||
|
|
||
|
furi_hal_power_disable_otg();
|
||
|
|
||
|
view_port_enabled_set(view_port, false);
|
||
|
gui_remove_view_port(gui, view_port);
|
||
|
furi_record_close("gui");
|
||
|
view_port_free(view_port);
|
||
|
furi_message_queue_free(event_queue);
|
||
|
delete_mutex(&coleco_mutex);
|
||
|
coleco_free(coleco);
|
||
|
return 0;
|
||
|
}
|