Flipper/Applications/Official/DEV_FW/source/xMasterX/rc2014_coleco/coleco.c
2023-01-25 23:52:38 -08:00

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;
}