src/sys/dev/usb/uoak_subr.c

361 lines
8.6 KiB
C

/* $OpenBSD: uoak_subr.c,v 1.11 2024/05/23 03:21:09 jsg Exp $ */
/*
* Copyright (c) 2012 Yojiro UO <yuo@nui.org>
*
* Permission to use, copy, modify, and distribute this software for any
* purpose with or without fee is hereby granted, provided that the above
* copyright notice and this permission notice appear in all copies.
*
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCAIMS ALL WARRANTIES
* WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
* ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
* WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
* ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
* OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
/* TORADEX OAK series sensors: common functions */
/* http://developer.toradex.com/files/toradex-dev/uploads/media/Oak/Oak_ProgrammingGuide.pdf */
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/device.h>
#include <sys/sensors.h>
#include <dev/usb/usb.h>
#include <dev/usb/usbhid.h>
#include <dev/usb/usbdi.h>
#include <dev/usb/usbdi_util.h>
#include <dev/usb/uhidev.h>
#include "uoak.h"
#define UOAK_RETRY_DELAY 100 /* 100ms, XXX too long? */
#define UOAK_RESPONSE_DELAY 10 /* 10ms, XXX too short? */
/*
* basic procedure to issue command to the OAK device.
* 1) check the device is ready to accept command.
* if a report of a FEATURE_REPORT request is not start 0xff,
* wait for a while, and retry till the response start with 0xff.
* 2) issue command. (set or get)
* 3) if the command will response, wait for a while, and issue
* FEATURE_REPORT. leading 0xff indicate the response is valid.
* if the first byte is not 0xff, retry.
*/
int
uoak_check_device_ready(struct uoak_softc *sc)
{
int actlen;
actlen = uhidev_get_report(sc->sc_hdev->sc_parent, UHID_FEATURE_REPORT,
sc->sc_hdev->sc_report_id, &sc->sc_buf, sc->sc_flen);
if (actlen != sc->sc_flen)
return EIO;
if (sc->sc_buf[0] != 0xff)
return -1;
return 0;
}
int
uoak_set_cmd(struct uoak_softc *sc)
{
int actlen;
sc->sc_rcmd.dir = OAK_SET;
while (uoak_check_device_ready(sc) < 0)
usbd_delay_ms(sc->sc_udev, UOAK_RETRY_DELAY);
actlen = uhidev_set_report(sc->sc_hdev->sc_parent, UHID_FEATURE_REPORT,
sc->sc_hdev->sc_report_id, &sc->sc_rcmd, sc->sc_flen);
if (actlen != sc->sc_flen)
return EIO;
return 0;
}
int
uoak_get_cmd(struct uoak_softc *sc)
{
int actlen;
sc->sc_rcmd.dir = OAK_GET;
/* check the device is ready to request */
while (uoak_check_device_ready(sc) < 0)
usbd_delay_ms(sc->sc_udev, UOAK_RETRY_DELAY);
/* issue request */
actlen = uhidev_set_report(sc->sc_hdev->sc_parent, UHID_FEATURE_REPORT,
sc->sc_hdev->sc_report_id, &sc->sc_rcmd, sc->sc_flen);
if (actlen != sc->sc_flen)
return EIO;
/* wait till the device ready to return the request */
while (uoak_check_device_ready(sc) < 0)
usbd_delay_ms(sc->sc_udev, UOAK_RESPONSE_DELAY);
return 0;
}
/*
* Functions to access device configurations.
* OAK sensor have some storages to store its configuration.
* (RAM, FLASH and others)
*/
int
uoak_get_device_name(struct uoak_softc *sc, enum uoak_target target)
{
memset(&sc->sc_rcmd, 0, sizeof(struct uoak_rcmd));
sc->sc_rcmd.target = target;
sc->sc_rcmd.datasize = 0x15;
USETW(&sc->sc_rcmd.cmd, OAK_CMD_DEVNAME);
if (uoak_get_cmd(sc) < 0)
return EIO;
strlcpy(sc->sc_config[target].devname, sc->sc_buf+1,
sizeof(sc->sc_config[target].devname));
return 0;
}
int
uoak_get_report_mode(struct uoak_softc *sc, enum uoak_target target)
{
memset(&sc->sc_rcmd, 0, sizeof(struct uoak_rcmd));
sc->sc_rcmd.target = target;
sc->sc_rcmd.datasize = 0x1;
USETW(&sc->sc_rcmd.cmd, OAK_CMD_REPORTMODE);
if (uoak_get_cmd(sc) < 0)
return EIO;
sc->sc_config[target].report_mode = sc->sc_buf[1];
return 0;
}
int
uoak_get_report_rate(struct uoak_softc *sc, enum uoak_target target)
{
uint16_t result;
memset(&sc->sc_rcmd, 0, sizeof(struct uoak_rcmd));
sc->sc_rcmd.target = target;
sc->sc_rcmd.datasize = 0x2;
USETW(&sc->sc_rcmd.cmd, OAK_CMD_REPORTRATE);
if (uoak_get_cmd(sc) < 0)
return EIO;
result = (sc->sc_buf[2] << 8) + sc->sc_buf[1];
sc->sc_config[target].report_rate = result;
return 0;
}
int
uoak_get_sample_rate(struct uoak_softc *sc, enum uoak_target target)
{
uint16_t result;
memset(&sc->sc_rcmd, 0, sizeof(struct uoak_rcmd));
sc->sc_rcmd.target = target;
sc->sc_rcmd.datasize = 0x2;
USETW(&sc->sc_rcmd.cmd, OAK_CMD_SAMPLERATE);
if (uoak_get_cmd(sc) < 0)
return EIO;
result = (sc->sc_buf[2] << 8) + sc->sc_buf[1];
sc->sc_config[target].sample_rate = result;
return 0;
}
int
uoak_set_sample_rate(struct uoak_softc *sc, enum uoak_target target, int rate)
{
memset(&sc->sc_rcmd, 0, sizeof(struct uoak_rcmd));
sc->sc_rcmd.target = target;
sc->sc_rcmd.datasize = 0x2;
USETW(&sc->sc_rcmd.cmd, OAK_CMD_SAMPLERATE);
#if 0
sc->sc_rcmd.val[0] = (uint8_t)(rate & 0xff);
sc->sc_rcmd.val[1] = (uint8_t)((rate >> 8) & 0xff)
#else
USETW(sc->sc_rcmd.val, rate);
#endif
if (uoak_set_cmd(sc) < 0)
return EIO;
return 0;
}
/*
* LED I/O
*/
int
uoak_led_status(struct uoak_softc *sc, enum uoak_target target, uint8_t *mode)
{
memset(&sc->sc_rcmd, 0, sizeof(struct uoak_rcmd));
sc->sc_rcmd.target = target;
sc->sc_rcmd.datasize = 0x1;
USETW(&sc->sc_rcmd.cmd, OAK_CMD_LEDMODE);
if (uoak_get_cmd(sc) < 0)
return EIO;
*mode = sc->sc_buf[1];
return 0;
}
int
uoak_led_ctrl(struct uoak_softc *sc, enum uoak_target target, uint8_t mode)
{
memset(&sc->sc_rcmd, 0, sizeof(struct uoak_rcmd));
sc->sc_rcmd.target = target;
sc->sc_rcmd.datasize = 0x1;
USETW(&sc->sc_rcmd.cmd, OAK_CMD_LEDMODE);
sc->sc_rcmd.val[0] = mode;
return uoak_set_cmd(sc);
}
/* device setting: query and pretty print */
void
uoak_get_devinfo(struct uoak_softc *sc)
{
/* get device serial# */
usbd_fill_deviceinfo(sc->sc_udev, &sc->sc_udi);
}
void
uoak_get_setting(struct uoak_softc *sc, enum uoak_target target)
{
/* get device level */
(void)uoak_get_device_name(sc, target);
/* get global sensor configuration */
(void)uoak_get_report_mode(sc, target);
(void)uoak_get_sample_rate(sc, target);
(void)uoak_get_report_rate(sc, target);
/* get device specific information */
if (sc->sc_methods->dev_setting != NULL)
sc->sc_methods->dev_setting(sc->sc_parent, target);
}
void
uoak_print_devinfo(struct uoak_softc *sc)
{
printf(": serial %s", sc->sc_udi.udi_serial);
}
void
uoak_print_setting(struct uoak_softc *sc, enum uoak_target target)
{
switch (sc->sc_config[target].report_mode) {
case OAK_REPORTMODE_AFTERSAMPLING:
printf(" sampling %dms",
sc->sc_config[target].sample_rate);
break;
case OAK_REPORTMODE_AFTERCHANGE:
printf(" reports changes");
break;
case OAK_REPORTMODE_FIXEDRATE:
printf(" rate %dms",
sc->sc_config[target].report_rate);
break;
default:
printf(" unknown sampling");
break;
}
/* print device specific information */
if (sc->sc_methods->dev_print != NULL)
sc->sc_methods->dev_print(sc->sc_parent, target);
printf("\n");
}
void
uoak_sensor_attach(struct uoak_softc *sc, struct uoak_sensor *s,
enum sensor_type type)
{
if (s == NULL)
return;
s->avg.type = type;
s->max.type = type;
s->min.type = type;
s->avg.flags |= SENSOR_FINVALID;
s->max.flags |= SENSOR_FINVALID;
s->min.flags |= SENSOR_FINVALID;
(void)snprintf(s->avg.desc, sizeof(s->avg.desc),
"avg(#%s)", sc->sc_udi.udi_serial);
(void)snprintf(s->max.desc, sizeof(s->max.desc),
"max(#%s)", sc->sc_udi.udi_serial);
(void)snprintf(s->min.desc, sizeof(s->min.desc),
"min(#%s)", sc->sc_udi.udi_serial);
sensor_attach(sc->sc_sensordev, &s->avg);
sensor_attach(sc->sc_sensordev, &s->max);
sensor_attach(sc->sc_sensordev, &s->min);
}
void
uoak_sensor_detach(struct uoak_softc *sc, struct uoak_sensor *s)
{
if (s == NULL)
return;
sensor_attach(sc->sc_sensordev, &s->avg);
sensor_attach(sc->sc_sensordev, &s->max);
sensor_attach(sc->sc_sensordev, &s->min);
}
void
uoak_sensor_update(struct uoak_sensor *s, int val)
{
if (s == NULL)
return;
/* reset */
if (s->count == 0) {
s->vmax = s->vmin = s->vavg = val;
s->count++;
return;
}
/* update min/max */
if (val > s->vmax)
s->vmax = val;
else if (val < s->vmin)
s->vmin = val;
/* calc average */
s->vavg = (s->vavg * s->count + val) / (s->count + 1);
s->count++;
}
void
uoak_sensor_refresh(struct uoak_sensor *s, int mag, int offset)
{
if (s == NULL)
return;
/* update value */
s->avg.value = s->vavg * mag + offset;
s->max.value = s->vmax * mag + offset;
s->min.value = s->vmin * mag + offset;
/* update flag */
s->avg.flags &= ~SENSOR_FINVALID;
s->max.flags &= ~SENSOR_FINVALID;
s->min.flags &= ~SENSOR_FINVALID;
s->count = 0;
}