forked from Dniel97/segatools
161 lines
4.1 KiB
C
161 lines
4.1 KiB
C
|
#include <windows.h>
|
||
|
|
||
|
#include <process.h>
|
||
|
#include <stdbool.h>
|
||
|
#include <stdint.h>
|
||
|
|
||
|
#include "mu3io/leddata.h"
|
||
|
#include "mu3io/pipeimpl.h"
|
||
|
|
||
|
static bool mu3_pipe_update[LED_BOARDS_TOTAL];
|
||
|
|
||
|
// incoming data is copied into these to ensure it isn't written during output
|
||
|
static struct _ongeki_led_data_buf_t mu3_pipe_write_buf[LED_BOARDS_TOTAL];
|
||
|
static HANDLE mu3_pipe_write_mutex;
|
||
|
|
||
|
static HRESULT mu3_pipe_create(LPHANDLE hPipe, LPCWSTR lpszPipename, DWORD dwBufSize)
|
||
|
{
|
||
|
*hPipe = INVALID_HANDLE_VALUE;
|
||
|
|
||
|
*hPipe = CreateNamedPipeW(
|
||
|
lpszPipename, // pipe name
|
||
|
PIPE_ACCESS_OUTBOUND, // read/write access
|
||
|
PIPE_TYPE_BYTE | // byte type pipe
|
||
|
PIPE_WAIT, // blocking mode
|
||
|
PIPE_UNLIMITED_INSTANCES, // max. instances
|
||
|
dwBufSize, // output buffer size
|
||
|
0, // input buffer size
|
||
|
0, // client time-out
|
||
|
NULL); // default security attribute
|
||
|
|
||
|
if (*hPipe == INVALID_HANDLE_VALUE)
|
||
|
{
|
||
|
return E_FAIL;
|
||
|
}
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
static HRESULT mu3_pipe_write(HANDLE hPipe, LPCVOID lpBuffer, DWORD dwSize)
|
||
|
{
|
||
|
DWORD cbWritten = 0;
|
||
|
|
||
|
bool fSuccess = WriteFile(
|
||
|
hPipe,
|
||
|
lpBuffer,
|
||
|
dwSize,
|
||
|
&cbWritten,
|
||
|
NULL);
|
||
|
|
||
|
if (!fSuccess || cbWritten != dwSize)
|
||
|
{
|
||
|
DWORD last_err = GetLastError();
|
||
|
return (last_err == ERROR_BROKEN_PIPE) ? E_ABORT : E_FAIL;
|
||
|
}
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
static unsigned int __stdcall mu3_io_led_pipe_thread_proc(void *ctx)
|
||
|
{
|
||
|
HANDLE hPipe;
|
||
|
LPCWSTR lpszPipename = L"\\\\.\\pipe\\ongeki_led";
|
||
|
|
||
|
while (true)
|
||
|
{
|
||
|
hPipe = INVALID_HANDLE_VALUE;
|
||
|
|
||
|
if (mu3_pipe_create(&hPipe, lpszPipename, LED_OUTPUT_TOTAL_SIZE_MAX) != S_OK)
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// wait for a connection to the pipe
|
||
|
bool fConnected = ConnectNamedPipe(hPipe, NULL) ?
|
||
|
true : (GetLastError() == ERROR_PIPE_CONNECTED);
|
||
|
|
||
|
while (fConnected)
|
||
|
{
|
||
|
if (WaitForSingleObject(mu3_pipe_write_mutex, INFINITE) != WAIT_OBJECT_0)
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
for (int i = 0; i < LED_BOARDS_TOTAL; i++) {
|
||
|
if (mu3_pipe_update[i])
|
||
|
{
|
||
|
HRESULT result = mu3_pipe_write(
|
||
|
hPipe,
|
||
|
&mu3_pipe_write_buf[i],
|
||
|
LED_OUTPUT_HEADER_SIZE + mu3_pipe_write_buf[i].data_len);
|
||
|
|
||
|
if (result != S_OK)
|
||
|
{
|
||
|
//if (result == E_ABORT)
|
||
|
//{
|
||
|
fConnected = false;
|
||
|
//}
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
mu3_pipe_update[i] = false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ReleaseMutex(mu3_pipe_write_mutex);
|
||
|
}
|
||
|
|
||
|
FlushFileBuffers(hPipe);
|
||
|
DisconnectNamedPipe(hPipe);
|
||
|
CloseHandle(hPipe);
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
HRESULT mu3_led_pipe_init()
|
||
|
{
|
||
|
mu3_pipe_write_mutex = CreateMutex(
|
||
|
NULL, // default security attributes
|
||
|
FALSE, // initially not owned
|
||
|
NULL); // unnamed mutex
|
||
|
|
||
|
if (mu3_pipe_write_mutex == NULL)
|
||
|
{
|
||
|
return E_FAIL;
|
||
|
}
|
||
|
|
||
|
// clear out update bools
|
||
|
for (int i = 0; i < LED_BOARDS_TOTAL; i++) {
|
||
|
mu3_pipe_update[i] = false;
|
||
|
}
|
||
|
|
||
|
_beginthreadex(
|
||
|
NULL,
|
||
|
0,
|
||
|
mu3_io_led_pipe_thread_proc,
|
||
|
0,
|
||
|
0,
|
||
|
NULL);
|
||
|
|
||
|
return S_OK;
|
||
|
}
|
||
|
|
||
|
void mu3_led_pipe_update(struct _ongeki_led_data_buf_t* data)
|
||
|
{
|
||
|
if (data->board > 1)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (WaitForSingleObject(mu3_pipe_write_mutex, INFINITE) != WAIT_OBJECT_0)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
memcpy(&mu3_pipe_write_buf[data->board], data, sizeof(struct _ongeki_led_data_buf_t));
|
||
|
mu3_pipe_update[data->board] = true;
|
||
|
|
||
|
ReleaseMutex(mu3_pipe_write_mutex);
|
||
|
}
|