pcsensor-temper/pcsensor.c

478 lines
15 KiB
C
Raw Normal View History

2012-09-13 20:03:51 +02:00
/*
* pcsensor.c by Juan Carlos Perez (c) 2011 (cray@isp-sl.com)
* based on Temper.c by Robert Kavaler (c) 2009 (relavak.com)
* All rights reserved.
*
* 2011/08/30 Thanks to EdorFaus: bugfix to support negative temperatures
2017-08-31 09:57:03 +09:00
* 2017/08/30 Improved by K.Cima: changed libusb-0.1 -> libusb-1.0
* https://github.com/shakemid/pcsensor
2012-09-13 20:03:51 +02:00
*
* Temper driver for linux. This program can be compiled either as a library
* or as a standalone program (-DUNIT_TEST). The driver will work with some
* TEMPer usb devices from RDing (www.PCsensor.com).
*
* 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.
*
* THIS SOFTWARE IS PROVIDED BY Juan Carlos Perez ''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 Robert kavaler 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.
*
*/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
2017-08-31 11:08:33 +09:00
#include <ctype.h>
2012-09-13 20:03:51 +02:00
#include <time.h>
2017-08-31 11:10:11 +09:00
#include <string.h>
#include <strings.h>
2012-09-13 20:03:51 +02:00
#include <errno.h>
#include <signal.h>
#include <libusb.h>
2012-09-13 20:03:51 +02:00
2017-09-15 15:42:11 +09:00
#define VERSION "1.2.0"
2012-09-13 20:03:51 +02:00
2017-09-15 15:42:11 +09:00
/* TEMPer type definition */
2017-09-15 17:20:07 +09:00
typedef struct {
2017-09-15 15:42:11 +09:00
const int vendor_id;
const int product_id;
const char product_name[256];
const int check_product_name; // if set, check product name in forward match
const int has_sensor; // number of temperature sensor
const int has_humid; // flag for humidity sensor
void (*decode_func)();
2017-09-15 17:09:36 +09:00
} temper_type_t;
2017-09-15 17:20:07 +09:00
typedef struct {
2017-09-15 17:09:36 +09:00
libusb_device_handle *handle;
temper_type_t *type;
} temper_device_t;
2017-09-15 15:42:11 +09:00
void decode_answer_fm75();
void decode_answer_sht1x();
2017-09-15 17:09:36 +09:00
#define TEMPER_TYPES 3
temper_type_t tempers[TEMPER_TYPES] = {
2017-09-15 15:42:11 +09:00
{ 0x0c45, 0x7401, "TEMPer2", 1, 2, 0, decode_answer_fm75 }, // TEMPer2* eg. TEMPer2V1.3
{ 0x0c45, 0x7401, "TEMPer1", 0, 1, 0, decode_answer_fm75 }, // other 0c45:7401 eg. TEMPerV1.4
{ 0x0c45, 0x7402, "TEMPerHUM", 0, 1, 1, decode_answer_sht1x },
};
/* global variables */
2012-09-13 20:03:51 +02:00
2017-09-15 17:09:36 +09:00
#define MAX_DEV 8
2012-09-13 20:03:51 +02:00
#define INTERFACE1 0x00
#define INTERFACE2 0x01
const static int reqIntLen=8;
const static int endpoint_Int_in=0x82; /* endpoint 0x81 address for IN */
const static int timeout=5000; /* timeout in ms */
2017-08-31 17:16:47 +09:00
const static char uTemperature[] = { 0x01, 0x80, 0x33, 0x01, 0x00, 0x00, 0x00, 0x00 };
2012-09-13 20:03:51 +02:00
const static char uIni1[] = { 0x01, 0x82, 0x77, 0x01, 0x00, 0x00, 0x00, 0x00 };
const static char uIni2[] = { 0x01, 0x86, 0xff, 0x01, 0x00, 0x00, 0x00, 0x00 };
static int bsalir=1;
static int debug=0;
static int seconds=5;
2017-09-14 11:56:09 +09:00
static int formato=1; //Celsius
2012-09-13 20:03:51 +02:00
2017-08-31 16:01:57 +09:00
static libusb_context *ctx = NULL;
2012-09-13 20:03:51 +02:00
2017-09-15 15:42:11 +09:00
/* functions */
2012-09-13 20:03:51 +02:00
void bad(const char *why) {
2017-08-31 09:56:15 +09:00
fprintf(stderr,"Fatal error> %s\n",why);
exit(17);
2012-09-13 20:03:51 +02:00
}
void usb_detach(libusb_device_handle *lvr_winusb, int iInterface) {
2017-08-31 09:56:15 +09:00
int ret;
ret = libusb_detach_kernel_driver(lvr_winusb, iInterface);
if(ret) {
if(errno == ENODATA) {
if(debug) {
2017-09-15 20:22:47 +09:00
fprintf(stderr, "Device already detached\n");
2017-08-31 09:56:15 +09:00
}
} else {
if(debug) {
2017-09-15 20:22:47 +09:00
fprintf(stderr, "Detach failed: %s[%d]\n", strerror(errno), errno);
fprintf(stderr, "Continuing anyway\n");
2017-08-31 09:56:15 +09:00
}
}
} else {
if(debug) {
2017-09-15 20:22:47 +09:00
fprintf(stderr, "detach successful\n");
2017-08-31 09:56:15 +09:00
}
}
2012-09-13 20:03:51 +02:00
}
2017-09-15 17:09:36 +09:00
int find_lvr_winusb(temper_device_t *devices) {
2017-09-15 15:42:11 +09:00
int i, j, s, cnt, numdev;
2017-08-31 16:01:57 +09:00
libusb_device **devs;
//handle = libusb_open_device_with_vid_pid(ctx, VENDOR_ID, PRODUCT_ID);
cnt = libusb_get_device_list(ctx, &devs);
if (cnt < 1) {
fprintf(stderr, "Could not find USB device: %d\n", cnt);
}
numdev = 0;
for (i = 0; i < cnt && numdev < MAX_DEV; i++) {
struct libusb_device_descriptor desc;
if ((s = libusb_get_device_descriptor(devs[i], &desc)) < 0) {
fprintf(stderr, "Could not get USB device descriptor: %d\n", s);
continue;
}
2017-09-15 15:42:11 +09:00
for (j = 0; j < TEMPER_TYPES; j++) {
if (desc.idVendor == tempers[j].vendor_id && desc.idProduct == tempers[j].product_id) {
unsigned char bus, addr, descmanu[256], descprod[256], descseri[256];
2017-09-15 15:42:11 +09:00
2017-09-15 12:42:48 +09:00
bus = libusb_get_bus_number(devs[i]);
addr = libusb_get_device_address(devs[i]);
2017-09-15 15:42:11 +09:00
2017-09-15 17:09:36 +09:00
if ((s = libusb_open(devs[i], &devices[numdev].handle)) < 0) {
2017-09-15 15:42:11 +09:00
fprintf(stderr, "Could not open USB device: %d\n", s);
continue;
}
2017-09-15 17:09:36 +09:00
libusb_get_string_descriptor_ascii(devices[numdev].handle, desc.iManufacturer, descmanu, 256);
libusb_get_string_descriptor_ascii(devices[numdev].handle, desc.iProduct, descprod, 256);
libusb_get_string_descriptor_ascii(devices[numdev].handle, desc.iSerialNumber, descseri, 256);
2017-09-14 23:10:36 +09:00
2017-09-15 15:42:11 +09:00
if (tempers[j].check_product_name) {
if (strncmp((const char*)descprod, tempers[j].product_name, strlen(tempers[j].product_name)) == 0) {
2017-09-15 17:09:36 +09:00
devices[numdev].type = &tempers[j];
2017-09-15 15:42:11 +09:00
}
else {
// vid and pid match, but product name unmatch
2017-09-15 17:09:36 +09:00
libusb_close(devices[numdev].handle);
2017-09-15 15:42:11 +09:00
continue;
}
}
else {
2017-09-15 17:09:36 +09:00
devices[numdev].type = &tempers[j];
2017-09-15 15:42:11 +09:00
}
if (debug) {
2017-09-15 20:22:47 +09:00
fprintf(stderr, "lvr_winusb with Bus:%03d Addr:%03d VendorID:%04x ProductID:%04x Manufacturer:%s Product:%s Serial:%s found.\n",
bus, addr, desc.idVendor, desc.idProduct, descmanu, descprod, descseri);
2017-09-15 15:42:11 +09:00
}
numdev++;
}
2017-08-31 16:01:57 +09:00
}
}
libusb_free_device_list(devs, 1);
return numdev;
}
2017-09-15 17:09:36 +09:00
int setup_libusb_access(temper_device_t *devices) {
2017-09-15 15:42:11 +09:00
int i;
int numdev;
2017-08-31 09:56:15 +09:00
libusb_init(&ctx);
2012-09-13 20:03:51 +02:00
2017-08-31 09:56:15 +09:00
if(debug) {
libusb_set_debug(ctx, 4); //LIBUSB_LOG_LEVEL_DEBUG
2017-08-31 09:56:15 +09:00
} else {
libusb_set_debug(ctx, 0); //LIBUSB_LOG_LEVEL_NONE
2017-08-31 09:56:15 +09:00
}
2012-09-13 20:03:51 +02:00
2017-09-15 17:09:36 +09:00
if((numdev = find_lvr_winusb(devices)) < 1) {
2017-08-31 16:01:57 +09:00
fprintf(stderr, "Couldn't find the USB device, Exiting: %d\n", numdev);
return -1;
2017-08-31 09:56:15 +09:00
}
2012-09-13 20:03:51 +02:00
2017-08-31 16:01:57 +09:00
for (i = 0; i < numdev; i++) {
2017-09-15 17:09:36 +09:00
usb_detach(devices[i].handle, INTERFACE1);
usb_detach(devices[i].handle, INTERFACE2);
2012-09-13 20:03:51 +02:00
2017-09-15 17:09:36 +09:00
if (libusb_set_configuration(devices[i].handle, 0x01) < 0) {
2017-08-31 16:01:57 +09:00
fprintf(stderr, "Could not set configuration 1\n");
return -1;
}
2012-09-13 20:03:51 +02:00
2017-08-31 16:01:57 +09:00
// Microdia tiene 2 interfaces
int s;
2017-09-15 17:09:36 +09:00
if ((s = libusb_claim_interface(devices[i].handle, INTERFACE1)) < 0) {
2017-08-31 16:01:57 +09:00
fprintf(stderr, "Could not claim interface. Error:%d\n", s);
return -1;
}
2012-09-13 20:03:51 +02:00
2017-09-15 17:09:36 +09:00
if ((s = libusb_claim_interface(devices[i].handle, INTERFACE2)) < 0) {
2017-08-31 16:01:57 +09:00
fprintf(stderr, "Could not claim interface. Error:%d\n", s);
return -1;
}
2017-08-31 09:56:15 +09:00
}
2012-09-13 20:03:51 +02:00
2017-08-31 16:01:57 +09:00
return numdev;
2012-09-13 20:03:51 +02:00
}
void ini_control_transfer(libusb_device_handle *dev) {
2012-09-13 20:03:51 +02:00
int r,i;
char question[] = { 0x01,0x01 };
2017-08-31 11:08:33 +09:00
r = libusb_control_transfer(dev, 0x21, 0x09, 0x0201, 0x00, (unsigned char *) question, 2, timeout);
2017-09-13 16:55:32 +09:00
if(r < 0) {
2017-08-31 09:56:15 +09:00
perror("USB control write"); bad("USB write failed");
2012-09-13 20:03:51 +02:00
}
if(debug) {
2017-09-15 20:22:47 +09:00
for (i=0;i<reqIntLen; i++) fprintf(stderr, "%02x ",question[i] & 0xFF);
fprintf(stderr, "\n");
2012-09-13 20:03:51 +02:00
}
}
void control_transfer(libusb_device_handle *dev, const char *pquestion) {
2012-09-13 20:03:51 +02:00
int r,i;
char question[reqIntLen];
memcpy(question, pquestion, sizeof question);
2017-08-31 11:08:33 +09:00
r = libusb_control_transfer(dev, 0x21, 0x09, 0x0200, 0x01, (unsigned char *) question, reqIntLen, timeout);
2017-09-13 16:55:32 +09:00
if(r < 0) {
2017-08-31 09:56:15 +09:00
perror("USB control write"); bad("USB write failed");
2012-09-13 20:03:51 +02:00
}
if(debug) {
2017-09-15 20:22:47 +09:00
for (i=0;i<reqIntLen; i++) fprintf(stderr, "%02x ",question[i] & 0xFF);
fprintf(stderr, "\n");
2012-09-13 20:03:51 +02:00
}
}
void interrupt_read(libusb_device_handle *dev, unsigned char *answer) {
int r,s,i;
2012-09-13 20:03:51 +02:00
bzero(answer, reqIntLen);
s = libusb_interrupt_transfer(dev, endpoint_Int_in, answer, reqIntLen, &r, timeout);
2017-09-13 16:55:32 +09:00
if(r != reqIntLen) {
2017-08-31 09:56:15 +09:00
fprintf(stderr, "USB read failed: %d\n", s);
perror("USB interrupt read"); bad("USB read failed");
2012-09-13 20:03:51 +02:00
}
if(debug) {
2017-09-15 20:22:47 +09:00
for (i=0;i<reqIntLen; i++) fprintf(stderr, "%02x ",answer[i] & 0xFF);
2012-09-13 20:03:51 +02:00
2017-09-15 20:22:47 +09:00
fprintf(stderr, "\n");
2012-09-13 20:03:51 +02:00
}
}
2017-09-15 20:44:33 +09:00
void cleanup_usb_devices(temper_device_t *devices, int numdev) {
int i;
for (i = 0; i < numdev; i++) {
libusb_release_interface(devices[i].handle, INTERFACE1);
libusb_release_interface(devices[i].handle, INTERFACE2);
libusb_close(devices[i].handle);
}
libusb_exit(ctx);
}
2012-09-13 20:03:51 +02:00
void ex_program(int sig) {
2017-08-31 09:56:15 +09:00
bsalir=1;
2012-09-13 20:03:51 +02:00
2017-08-31 09:56:15 +09:00
(void) signal(SIGINT, SIG_DFL);
2012-09-13 20:03:51 +02:00
}
2017-09-15 15:42:11 +09:00
/* decode funcs */
/* Thanks to https://github.com/edorfaus/TEMPered */
void decode_answer_fm75(unsigned char *answer, float *tempd, float *calibration) {
int buf;
// temp C internal
buf = ((signed char)answer[2] << 8) + (answer[3] & 0xFF);
2017-09-26 08:02:36 +09:00
if (debug) fprintf(stderr, "%d\n", buf);
2017-09-15 15:42:11 +09:00
tempd[0] = buf * (125.0 / 32000.0);
tempd[0] = tempd[0] * calibration[0] + calibration[1];
// temp C external
buf = ((signed char)answer[4] << 8) + (answer[5] & 0xFF);
2017-09-26 08:02:36 +09:00
if (debug) fprintf(stderr, "%d\n", buf);
2017-09-15 15:42:11 +09:00
tempd[1] = buf * (125.0 / 32000.0);
tempd[1] = tempd[1] * calibration[0] + calibration[1];
};
void decode_answer_sht1x(unsigned char *answer, float *tempd, float *calibration){
int buf;
// temp C
2017-09-26 08:02:36 +09:00
buf = ((signed char)answer[2] << 8) + (answer[3] & 0xFF);
if (debug) fprintf(stderr, "%d\n", buf);
2017-09-15 15:42:11 +09:00
tempd[0] = -39.7 + 0.01 * buf;
tempd[0] = tempd[0] * calibration[0] + calibration[1];
// relative humidity
2017-09-26 08:02:36 +09:00
buf = ((signed char)answer[4] << 8) + (answer[5] & 0xFF);
if (debug) fprintf(stderr, "%d\n", buf);
2017-09-15 15:42:11 +09:00
tempd[1] = -2.0468 + 0.0367 * buf - 1.5955e-6 * buf * buf;
2017-09-15 23:27:46 +09:00
tempd[1] = ( tempd[0] - 25 ) * ( 0.01 + 0.00008 * buf ) + tempd[1];
2017-09-15 23:25:06 +09:00
if (tempd[1] < 0) tempd[1] = 0;
if (tempd[1] > 99) tempd[1] = 100;
2017-09-15 15:42:11 +09:00
};
int main(int argc, char **argv) {
2017-09-15 17:09:36 +09:00
temper_device_t *devices;
2017-08-31 16:01:57 +09:00
int numdev,i;
unsigned char *answer;
2017-09-15 15:42:11 +09:00
float tempd[2];
float calibration[2] = { 1, 0 }; //scale, offset
2017-09-14 11:56:09 +09:00
char strdate[20];
2017-08-31 09:56:15 +09:00
int c;
struct tm *local;
time_t t;
2017-09-15 15:42:11 +09:00
while ((c = getopt(argc, argv, "vcfl::a:h")) != -1)
2017-08-31 09:56:15 +09:00
switch (c)
{
case 'v':
debug = 1;
break;
2017-09-15 12:19:58 +09:00
case 'c':
formato=1; //Celsius
break;
2017-08-31 09:56:15 +09:00
case 'f':
formato=2; //Fahrenheit
break;
case 'l':
if (optarg!=NULL){
2017-09-15 15:42:11 +09:00
if (!(sscanf(optarg,"%i",&seconds) == 1)) {
2017-08-31 09:56:15 +09:00
fprintf (stderr, "Error: '%s' is not numeric.\n", optarg);
exit(EXIT_FAILURE);
} else {
bsalir = 0;
break;
}
} else {
bsalir = 0;
seconds = 5;
break;
}
2017-09-15 15:42:11 +09:00
case 'a':
if (!(sscanf(optarg,"%f:%f", &calibration[0], &calibration[1]) == 2)) {
fprintf (stderr, "Error: '%s' is not numeric.\n", optarg);
exit(EXIT_FAILURE);
}
break;
2017-08-31 09:56:15 +09:00
case '?':
case 'h':
printf("pcsensor version %s\n",VERSION);
2017-08-31 17:16:47 +09:00
printf(" Available options:\n");
printf(" -h help\n");
printf(" -v verbose\n");
printf(" -l[n] loop every 'n' seconds, default value is 5s\n");
2017-09-15 17:21:34 +09:00
printf(" -a scale:offset set values for calibration TempC*scale+offset eg. 1.02:-0.55 \n");
2017-09-15 12:19:58 +09:00
printf(" -c output in Celsius (default)\n");
printf(" -f output in Fahrenheit\n");
2017-08-31 09:56:15 +09:00
exit(EXIT_FAILURE);
default:
if (isprint (optopt))
fprintf (stderr, "Unknown option `-%c'.\n", optopt);
else
2017-08-31 16:01:57 +09:00
fprintf (stderr, "Unknown option character `\\x%x'.\n", optopt);
2017-08-31 09:56:15 +09:00
exit(EXIT_FAILURE);
}
2012-09-13 20:03:51 +02:00
2017-08-31 09:56:15 +09:00
if (optind < argc) {
2012-09-13 20:03:51 +02:00
fprintf(stderr, "Non-option ARGV-elements, try -h for help.\n");
exit(EXIT_FAILURE);
2017-08-31 09:56:15 +09:00
}
2012-09-13 20:03:51 +02:00
2017-09-15 17:56:12 +09:00
devices = calloc(MAX_DEV, sizeof(temper_device_t));
2017-09-15 17:09:36 +09:00
if ((numdev = setup_libusb_access(devices)) < 1) {
2017-08-31 09:56:15 +09:00
exit(EXIT_FAILURE);
}
2012-09-13 20:03:51 +02:00
2017-08-31 09:56:15 +09:00
(void) signal(SIGINT, ex_program);
answer = calloc(reqIntLen, sizeof(unsigned char));
2017-09-14 09:56:38 +09:00
/* This looks unnecessary
2017-08-31 16:01:57 +09:00
for (i = 0; i < numdev; i++) {
2017-09-15 17:16:04 +09:00
ini_control_transfer(devices[i].handle);
2017-08-31 09:56:15 +09:00
2017-09-15 17:16:04 +09:00
control_transfer(devices[i].handle, uTemperature);
interrupt_read(devices[i].handle, answer);
2017-08-31 09:56:15 +09:00
2017-09-15 17:16:04 +09:00
control_transfer(devices[i].handle, uIni1);
interrupt_read(devices[i].handle, answer);
2017-08-31 09:56:15 +09:00
2017-09-15 17:16:04 +09:00
control_transfer(devices[i].handle, uIni2);
interrupt_read(devices[i].handle, answer);
interrupt_read(devices[i].handle, answer);
2017-08-31 16:01:57 +09:00
}
2017-09-14 09:56:38 +09:00
*/
2017-08-31 09:56:15 +09:00
do {
2017-08-31 16:01:57 +09:00
for (i = 0; i < numdev; i++) {
2017-09-15 17:56:12 +09:00
// get localtime
2017-08-31 16:01:57 +09:00
t = time(NULL);
local = localtime(&t);
2017-09-14 11:56:09 +09:00
sprintf(strdate, "%04d-%02d-%02dT%02d:%02d:%02d",
local->tm_year +1900,
local->tm_mon + 1,
local->tm_mday,
local->tm_hour,
local->tm_min,
local->tm_sec);
2017-09-15 17:56:12 +09:00
// get temperature
control_transfer(devices[i].handle, uTemperature);
interrupt_read(devices[i].handle, answer);
devices[i].type->decode_func(answer, tempd, calibration);
2017-09-15 15:42:11 +09:00
// print temperature
2017-09-15 19:14:30 +09:00
printf("%s\t%d\t%s\t%.2f %s\n",
strdate,
i,
devices[i].type->has_sensor == 2 ? "internal" : "temperature",
formato == 2 ? 9.0 / 5.0 * tempd[0] + 32.0 : tempd[0],
formato == 2 ? "F" : "C");
if (devices[i].type->has_sensor == 2) {
printf("%s\t%d\texternal\t%.2f %s\n",
strdate,
i,
formato == 2 ? 9.0 / 5.0 * tempd[1] + 32.0 : tempd[1],
formato == 2 ? "F" : "C");
2017-09-15 15:42:11 +09:00
}
// print humidity
2017-09-15 17:09:36 +09:00
if (devices[i].type->has_humid == 1) {
2017-09-15 15:42:11 +09:00
printf("%s\t%d\thumidity\t%.2f %%\n", strdate, i, tempd[1]);
2017-08-31 09:56:15 +09:00
}
2017-08-31 16:01:57 +09:00
}
2017-09-15 17:58:09 +09:00
if (!bsalir)
sleep(seconds);
2017-08-31 09:56:15 +09:00
} while (!bsalir);
2012-09-13 20:03:51 +02:00
2017-09-15 20:44:33 +09:00
cleanup_usb_devices(devices, numdev);
2017-09-15 20:23:03 +09:00
2017-08-31 09:56:15 +09:00
return 0;
2012-09-13 20:03:51 +02:00
}