forked from Hay1tsme/segatools
202 lines
4.8 KiB
C
202 lines
4.8 KiB
C
#include <windows.h>
|
|
|
|
#include <assert.h>
|
|
#include <stdbool.h>
|
|
#include <stdint.h>
|
|
|
|
#include "board/aime-dll.h"
|
|
#include "board/sg-led.h"
|
|
#include "board/sg-nfc.h"
|
|
#include "board/sg-reader.h"
|
|
|
|
#include "hook/iohook.h"
|
|
|
|
#include "hooklib/uart.h"
|
|
|
|
#include "util/dprintf.h"
|
|
#include "util/dump.h"
|
|
|
|
static HRESULT sg_reader_handle_irp(struct irp *irp);
|
|
static HRESULT sg_reader_handle_irp_locked(struct irp *irp);
|
|
static HRESULT sg_reader_nfc_poll(void *ctx);
|
|
static HRESULT sg_reader_nfc_get_aime_id(
|
|
void *ctx,
|
|
uint8_t *luid,
|
|
size_t luid_size);
|
|
static HRESULT sg_reader_nfc_get_felica_id(void *ctx, uint64_t *IDm);
|
|
static void sg_reader_led_set_color(void *ctx, uint8_t r, uint8_t g, uint8_t b);
|
|
|
|
static const struct sg_nfc_ops sg_reader_nfc_ops = {
|
|
.poll = sg_reader_nfc_poll,
|
|
.get_aime_id = sg_reader_nfc_get_aime_id,
|
|
.get_felica_id = sg_reader_nfc_get_felica_id,
|
|
};
|
|
|
|
static const struct sg_led_ops sg_reader_led_ops = {
|
|
.set_color = sg_reader_led_set_color,
|
|
};
|
|
|
|
static CRITICAL_SECTION sg_reader_lock;
|
|
static bool sg_reader_started;
|
|
static HRESULT sg_reader_start_hr;
|
|
static struct uart sg_reader_uart;
|
|
static uint8_t sg_reader_written_bytes[520];
|
|
static uint8_t sg_reader_readable_bytes[520];
|
|
static struct sg_nfc sg_reader_nfc;
|
|
static struct sg_led sg_reader_led;
|
|
|
|
HRESULT sg_reader_hook_init(
|
|
const struct aime_config *cfg,
|
|
unsigned int port_no,
|
|
unsigned int gen,
|
|
HINSTANCE self)
|
|
{
|
|
HRESULT hr;
|
|
|
|
assert(cfg != NULL);
|
|
assert(self != NULL);
|
|
|
|
if (!cfg->enable) {
|
|
return S_FALSE;
|
|
}
|
|
|
|
hr = aime_dll_init(&cfg->dll, self);
|
|
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
|
|
if (cfg->gen != 0) {
|
|
gen = cfg->gen;
|
|
}
|
|
|
|
if (gen < 1 || gen > 3) {
|
|
dprintf("NFC Assembly: Invalid reader generation: %u\n", gen);
|
|
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
sg_nfc_init(&sg_reader_nfc, 0x00, &sg_reader_nfc_ops, gen, NULL);
|
|
sg_led_init(&sg_reader_led, 0x08, &sg_reader_led_ops, gen, NULL);
|
|
|
|
InitializeCriticalSection(&sg_reader_lock);
|
|
|
|
if (!cfg->high_baudrate) {
|
|
sg_reader_uart.baud.BaudRate = 38400;
|
|
}
|
|
|
|
uart_init(&sg_reader_uart, port_no);
|
|
sg_reader_uart.written.bytes = sg_reader_written_bytes;
|
|
sg_reader_uart.written.nbytes = sizeof(sg_reader_written_bytes);
|
|
sg_reader_uart.readable.bytes = sg_reader_readable_bytes;
|
|
sg_reader_uart.readable.nbytes = sizeof(sg_reader_readable_bytes);
|
|
|
|
return iohook_push_handler(sg_reader_handle_irp);
|
|
}
|
|
|
|
static HRESULT sg_reader_handle_irp(struct irp *irp)
|
|
{
|
|
HRESULT hr;
|
|
|
|
assert(irp != NULL);
|
|
|
|
if (!uart_match_irp(&sg_reader_uart, irp)) {
|
|
return iohook_invoke_next(irp);
|
|
}
|
|
|
|
EnterCriticalSection(&sg_reader_lock);
|
|
hr = sg_reader_handle_irp_locked(irp);
|
|
LeaveCriticalSection(&sg_reader_lock);
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT sg_reader_handle_irp_locked(struct irp *irp)
|
|
{
|
|
HRESULT hr;
|
|
|
|
#if defined(LOG_NFC)
|
|
if (irp->op == IRP_OP_WRITE) {
|
|
dprintf("WRITE:\n");
|
|
dump_const_iobuf(&irp->write);
|
|
}
|
|
#endif
|
|
|
|
#if defined(LOG_NFC)
|
|
if (irp->op == IRP_OP_READ) {
|
|
dprintf("READ:\n");
|
|
dump_iobuf(&sg_reader_uart.readable);
|
|
}
|
|
#endif
|
|
|
|
if (irp->op == IRP_OP_OPEN) {
|
|
/* Unfortunately the card reader UART gets opened and closed
|
|
repeatedly */
|
|
|
|
if (!sg_reader_started) {
|
|
dprintf("NFC Assembly: Starting backend DLL\n");
|
|
hr = aime_dll.init();
|
|
|
|
sg_reader_started = true;
|
|
sg_reader_start_hr = hr;
|
|
|
|
if (FAILED(hr)) {
|
|
dprintf("NFC Assembly: Backend error: %x\n", (int) hr);
|
|
|
|
return hr;
|
|
}
|
|
} else {
|
|
hr = sg_reader_start_hr;
|
|
|
|
if (FAILED(hr)) {
|
|
return hr;
|
|
}
|
|
}
|
|
}
|
|
|
|
hr = uart_handle_irp(&sg_reader_uart, irp);
|
|
|
|
if (FAILED(hr) || irp->op != IRP_OP_WRITE) {
|
|
return hr;
|
|
}
|
|
|
|
sg_nfc_transact(
|
|
&sg_reader_nfc,
|
|
&sg_reader_uart.readable,
|
|
sg_reader_uart.written.bytes,
|
|
sg_reader_uart.written.pos);
|
|
|
|
sg_led_transact(
|
|
&sg_reader_led,
|
|
&sg_reader_uart.readable,
|
|
sg_reader_uart.written.bytes,
|
|
sg_reader_uart.written.pos);
|
|
|
|
sg_reader_uart.written.pos = 0;
|
|
|
|
return hr;
|
|
}
|
|
|
|
static HRESULT sg_reader_nfc_poll(void *ctx)
|
|
{
|
|
return aime_dll.nfc_poll(0);
|
|
}
|
|
|
|
static HRESULT sg_reader_nfc_get_aime_id(
|
|
void *ctx,
|
|
uint8_t *luid,
|
|
size_t luid_size)
|
|
{
|
|
return aime_dll.nfc_get_aime_id(0, luid, luid_size);
|
|
}
|
|
|
|
static HRESULT sg_reader_nfc_get_felica_id(void *ctx, uint64_t *IDm)
|
|
{
|
|
return aime_dll.nfc_get_felica_id(0, IDm);
|
|
}
|
|
|
|
static void sg_reader_led_set_color(void *ctx, uint8_t r, uint8_t g, uint8_t b)
|
|
{
|
|
aime_dll.led_set_color(0, r, g, b);
|
|
}
|