123 lines
4 KiB
C
123 lines
4 KiB
C
/* * Copyright (c) 2023 Carl Klemm <carl@uvos.xyz>
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without modification,
|
|
* are permitted provided that the following conditions are met:
|
|
*
|
|
* * Redistributions of source code must retain the above copyright notice, this
|
|
* list of conditions and the following disclaimer.
|
|
* * Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation and/or
|
|
* other materials provided with the distribution.
|
|
* * Neither the name of %ORGANIZATION% nor the names of its contributors may be
|
|
* used to endorse or promote products derived from this software without specific
|
|
* prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#define _POSIX_C_SOURCE 199309L
|
|
#include "eismultiplexer.h"
|
|
#include "usbshm.h"
|
|
#include <stdlib.h>
|
|
#include <time.h>
|
|
|
|
static void usleep(uint64_t microseconds)
|
|
{
|
|
struct timespec ts;
|
|
ts.tv_sec = microseconds / 1000000;
|
|
ts.tv_nsec = (microseconds % 1000000) * 1000;
|
|
nanosleep(&ts, NULL);
|
|
}
|
|
|
|
static void dataCallback(uint8_t request, unsigned char* data, size_t length, void* user_data)
|
|
{
|
|
struct eismultiplexer* muliplexer = user_data;
|
|
|
|
if(length >= 1)
|
|
muliplexer->activeChannels = data[0];
|
|
|
|
sem_post(&muliplexer->readSem);
|
|
}
|
|
|
|
int eismultiplexer_connect(struct eismultiplexer* muliplexer, int serial)
|
|
{
|
|
int ret;
|
|
muliplexer->priv = malloc(sizeof(*muliplexer->priv));
|
|
if(!muliplexer->priv)
|
|
return -1;
|
|
|
|
ret = usbshm_init(muliplexer->priv, &dataCallback, muliplexer);
|
|
if(ret)
|
|
return -2;
|
|
|
|
ret = usbshm_open(muliplexer->priv, 0xfe17, 0x07dc , NULL);
|
|
if(ret)
|
|
return -3;
|
|
|
|
ret = sem_init(&muliplexer->readSem, 0, 0);
|
|
if(ret)
|
|
return -4;
|
|
return 0;
|
|
}
|
|
|
|
int eismultiplexer_connect_channel_exclusive(struct eismultiplexer* muliplexer, channel_t channel)
|
|
{
|
|
uint16_t wValue;
|
|
// we compile with fno-strict-aliasing
|
|
uint8_t* wValChar = (uint8_t*)&wValue;
|
|
wValChar[0] = channel;
|
|
wValChar[1] = 0;
|
|
int ret;
|
|
while((ret = usbshm_writeControlTransfer(muliplexer->priv, 2, NULL, 0, wValue, 0)) == USBSHM_ERROR_AGAIN)
|
|
usleep(1000000);
|
|
return ret;
|
|
}
|
|
|
|
int eismultiplexer_connect_channel(struct eismultiplexer* muliplexer, channel_t channel)
|
|
{
|
|
channel_t channels = eismultiplexer_get_connected(muliplexer);
|
|
channels |= channel;
|
|
return eismultiplexer_connect_channel_exclusive(muliplexer, channels);
|
|
}
|
|
|
|
int eismultiplexer_disconnect_channel(struct eismultiplexer* muliplexer, channel_t channel)
|
|
{
|
|
channel_t channels = eismultiplexer_get_connected(muliplexer);
|
|
channels &= ~channel;
|
|
return eismultiplexer_connect_channel_exclusive(muliplexer, channels);
|
|
}
|
|
|
|
channel_t eismultiplexer_get_connected(struct eismultiplexer* muliplexer)
|
|
{
|
|
usbshm_readControlTransfer(muliplexer->priv, 3, 1);
|
|
sem_wait(&muliplexer->readSem);
|
|
return muliplexer->activeChannels;
|
|
}
|
|
|
|
|
|
int eismultiplexer_set_led(struct eismultiplexer* muliplexer, bool on)
|
|
{
|
|
int ret;
|
|
while((ret = usbshm_writeControlTransfer(muliplexer->priv, on, NULL, 0, 0, 0)) == USBSHM_ERROR_AGAIN)
|
|
usleep(1000000);
|
|
return ret;
|
|
}
|
|
|
|
void eismultiplexer_disconnect(struct eismultiplexer* muliplexer)
|
|
{
|
|
usbshm_distroy(muliplexer->priv);
|
|
free(muliplexer->priv);
|
|
muliplexer->priv = NULL;
|
|
sem_destroy(&muliplexer->readSem);
|
|
}
|