#include #include #include #include #include "RC2014_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; }