mirror of
https://github.com/slsdetectorgroup/slsDetectorPackage.git
synced 2025-04-24 23:30:03 +02:00
435 lines
14 KiB
C
435 lines
14 KiB
C
#define _GNU_SOURCE // needed for strptime to be at the top
|
|
#include "common.h"
|
|
#include "clogger.h"
|
|
#include "sls/sls_detector_defs.h"
|
|
|
|
#include <libgen.h> // dirname
|
|
#include <string.h>
|
|
#include <unistd.h> // readlink
|
|
|
|
#define FLASH_CONVERTED_NAME "/var/tmp/tests.rawbin"
|
|
|
|
int ConvertToDifferentRange(int inputMin, int inputMax, int outputMin,
|
|
int outputMax, int inputValue, int *outputValue) {
|
|
LOG(logDEBUG1, (" Input Value: %d (Input:(%d - %d), Output:(%d - %d))\n",
|
|
inputValue, inputMin, inputMax, outputMin, outputMax));
|
|
|
|
// validate within bounds
|
|
// eg. MAX1932 range is v(60 - 200) to dac(255 - 1), here inputMin >
|
|
// inputMax (when dac to voltage)
|
|
int smaller = inputMin;
|
|
int bigger = inputMax;
|
|
if (smaller > bigger) {
|
|
smaller = inputMax;
|
|
bigger = inputMin;
|
|
}
|
|
if ((inputValue < smaller) || (inputValue > bigger)) {
|
|
LOG(logERROR, ("Input Value is outside bounds (%d to %d): %d\n",
|
|
smaller, bigger, inputValue));
|
|
*outputValue = -1;
|
|
return FAIL;
|
|
}
|
|
|
|
double value =
|
|
((double)(inputValue - inputMin) * (double)(outputMax - outputMin)) /
|
|
(double)(inputMax - inputMin) +
|
|
outputMin;
|
|
|
|
// double to integer conversion (if decimal places, round to integer)
|
|
if ((value - (int)value) > 0.0001) {
|
|
value += 0.5;
|
|
}
|
|
*outputValue = value;
|
|
|
|
LOG(logDEBUG1, (" Converted Output Value: %d\n", *outputValue));
|
|
return OK;
|
|
}
|
|
|
|
int getAbsPath(char *buf, size_t bufSize, char *fname) {
|
|
// get path of current binary
|
|
char path[bufSize];
|
|
memset(path, 0, bufSize);
|
|
ssize_t len = readlink("/proc/self/exe", path, bufSize - 1);
|
|
if (len < 0) {
|
|
LOG(logWARNING, ("Could not readlink current binary for %s\n", fname));
|
|
return FAIL;
|
|
}
|
|
path[len] = '\0';
|
|
|
|
// get dir path and attach config file name
|
|
char *dir = dirname(path);
|
|
memset(buf, 0, bufSize);
|
|
sprintf(buf, "%s/%s", dir, fname);
|
|
LOG(logDEBUG1, ("full path for %s: %s\n", fname, buf));
|
|
return OK;
|
|
}
|
|
|
|
int GetTimeFromString(char *buf, time_t *result) {
|
|
struct tm t;
|
|
if (NULL == strptime(buf, "%a %b %d %H:%M:%S %Z %Y", &t)) {
|
|
return FAIL;
|
|
}
|
|
*result = mktime(&t);
|
|
return OK;
|
|
}
|
|
|
|
void validate(int *ret, char *mess, int arg, int retval, char *modename,
|
|
enum numberMode nummode) {
|
|
if (*ret == OK && arg != GET_FLAG && retval != arg) {
|
|
*ret = FAIL;
|
|
if (nummode == HEX)
|
|
sprintf(mess, "Could not %s. Set 0x%x, but read 0x%x\n", modename,
|
|
arg, retval);
|
|
else
|
|
sprintf(mess, "Could not %s. Set %d, but read %d\n", modename, arg,
|
|
retval);
|
|
LOG(logERROR, (mess));
|
|
}
|
|
}
|
|
|
|
void validate64(int *ret, char *mess, int64_t arg, int64_t retval,
|
|
char *modename, enum numberMode nummode) {
|
|
if (*ret == OK && arg != GET_FLAG && retval != arg) {
|
|
*ret = FAIL;
|
|
if (nummode == HEX)
|
|
sprintf(mess, "Could not %s. Set 0x%llx, but read 0x%llx\n",
|
|
modename, (long long unsigned int)arg,
|
|
(long long unsigned int)retval);
|
|
else
|
|
sprintf(mess, "Could not %s. Set %lld, but read %lld\n", modename,
|
|
(long long unsigned int)arg,
|
|
(long long unsigned int)retval);
|
|
LOG(logERROR, (mess));
|
|
}
|
|
}
|
|
|
|
int getModuleIdInFile(int *ret, char *mess, char *fileName) {
|
|
const int fileNameSize = 128;
|
|
char fname[fileNameSize];
|
|
if (getAbsPath(fname, fileNameSize, fileName) == FAIL) {
|
|
*ret = FAIL;
|
|
strcpy(mess, "Could not find detid file\n");
|
|
LOG(logERROR, ("%s\n\n", mess));
|
|
return -1;
|
|
}
|
|
|
|
// open id file
|
|
FILE *fd = fopen(fname, "r");
|
|
if (fd == NULL) {
|
|
*ret = FAIL;
|
|
strcpy(mess, "Could not find detid file\n");
|
|
LOG(logERROR, ("%s\n\n", mess));
|
|
return -1;
|
|
}
|
|
LOG(logDEBUG1, ("Reading det id file %s\n", fileName));
|
|
|
|
// read line
|
|
const size_t len = 256;
|
|
char line[len];
|
|
memset(line, 0, len);
|
|
if (NULL == fgets(line, len, fd)) {
|
|
*ret = FAIL;
|
|
strcpy(mess, "Could not find detid file\n");
|
|
LOG(logERROR, ("%s\n\n", mess));
|
|
return -1;
|
|
}
|
|
// read id
|
|
int retval = 0;
|
|
if (sscanf(line, "%x", &retval) != 1) {
|
|
*ret = FAIL;
|
|
sprintf(mess,
|
|
"Could not scan det id from on-board server "
|
|
"id file. Line:[%s].\n",
|
|
line);
|
|
LOG(logERROR, ("%s\n\n", mess));
|
|
return -1;
|
|
}
|
|
LOG(logINFOBLUE, ("Module Id: 0x%x (File)\n", retval));
|
|
return retval;
|
|
}
|
|
|
|
int setModuleIdInFile(char *mess, int arg, char *fileName) {
|
|
LOG(logINFOBLUE, ("Setting Module Id: 0x%x (File)\n", arg));
|
|
|
|
const int fileNameSize = 128;
|
|
char fname[fileNameSize];
|
|
if (getAbsPath(fname, fileNameSize, fileName) == FAIL) {
|
|
strcpy(mess, "Could not find detid file\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
|
|
// open id file
|
|
FILE *fd = fopen(fname, "r");
|
|
if (fd == NULL) {
|
|
strcpy(mess, "Could not find detid file\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
LOG(logDEBUG1, ("Writing det id to file %s\n", fileName));
|
|
|
|
// write id
|
|
const size_t len = 256;
|
|
char line[len];
|
|
memset(line, 0, len);
|
|
sprintf(line, "%x", arg);
|
|
if (EOF == fputs(line, fd)) {
|
|
strcpy(mess, "Could not write to detid file\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
return OK;
|
|
}
|
|
|
|
int verifyChecksumFromBuffer(char *mess, char *clientChecksum, char *buffer,
|
|
ssize_t bytes) {
|
|
LOG(logINFO, ("\tVerifying Checksum...\n"));
|
|
MD5_CTX c;
|
|
if (!MD5_Init(&c)) {
|
|
strcpy(mess, "Unable to calculate checksum (MD5_Init)\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
if (!MD5_Update(&c, buffer, bytes)) {
|
|
strcpy(mess, "Unable to calculate checksum (MD5_Update)\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
return verifyChecksum(mess, clientChecksum, &c, "copied program");
|
|
}
|
|
|
|
int verifyChecksumFromFile(char *mess, char *clientChecksum, char *fname) {
|
|
LOG(logINFO, ("\tVerifying Checksum...\n"));
|
|
|
|
FILE *fp = fopen(fname, "r");
|
|
if (fp == NULL) {
|
|
sprintf(mess, "Unable to open %s in read mode to get checksum\n",
|
|
fname);
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
|
|
MD5_CTX c;
|
|
if (!MD5_Init(&c)) {
|
|
fclose(fp);
|
|
strcpy(mess, "Unable to calculate checksum (MD5_Init)\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
const int readUnitSize = 128;
|
|
char buf[readUnitSize];
|
|
ssize_t bytes = fread(buf, 1, readUnitSize, fp);
|
|
ssize_t totalBytesRead = bytes;
|
|
while (bytes > 0) {
|
|
if (!MD5_Update(&c, buf, bytes)) {
|
|
fclose(fp);
|
|
strcpy(mess, "Unable to calculate checksum (MD5_Update)\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
bytes = fread(buf, 1, readUnitSize, fp);
|
|
totalBytesRead += bytes;
|
|
}
|
|
LOG(logINFO, ("\tRead %lu bytes to calculate checksum\n", totalBytesRead));
|
|
fclose(fp);
|
|
return verifyChecksum(mess, clientChecksum, &c, "copied program");
|
|
}
|
|
/*
|
|
int verifyChecksumFromFlash(char *mess, char *clientChecksum, char *fname,
|
|
ssize_t fsize) {
|
|
LOG(logINFO, ("\tVerifying FlashChecksum...\n"));
|
|
|
|
FILE *fp = fopen(fname, "r");
|
|
if (fp == NULL) {
|
|
sprintf(mess, "Unable to open %s in read mode to get checksum\n",
|
|
fname);
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
|
|
FILE *flashfp = fopen(FLASH_CONVERTED_NAME, "w");
|
|
if (flashfp == NULL) {
|
|
sprintf(mess, "Unable to open %s in write mode to write checksum\n",
|
|
FLASH_CONVERTED_NAME);
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
|
|
MD5_CTX c;
|
|
if (!MD5_Init(&c)) {
|
|
fclose(fp);
|
|
strcpy(mess, "Unable to calculate checksum (MD5_Init)\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
const int readUnitSize = 128;
|
|
char buf[readUnitSize + 1];
|
|
ssize_t bytes = fread(buf, 1, readUnitSize + 1, fp);
|
|
// the first time, extra bytes are read
|
|
--bytes;
|
|
ssize_t totalBytesRead = bytes;
|
|
char lastByte = buf[0];
|
|
int oldProgress = 0;
|
|
|
|
while (bytes > 0) {
|
|
if (totalBytesRead <= 2 * bytes) {
|
|
LOG(logINFO, ("bytes:%d\n", bytes));
|
|
}
|
|
int progress = (int)(((double)(totalBytesRead) / fsize) * 100);
|
|
if (oldProgress != progress) {
|
|
printf("%d%%\r", progress);
|
|
fflush(stdout);
|
|
oldProgress = progress;
|
|
}
|
|
|
|
buf[0] = lastByte;
|
|
if (totalBytesRead <= 2 * bytes) {
|
|
for (int i = 0; i < 3; ++i) {
|
|
printf("%d:[0x%02x]\n", i, (uint8_t)buf[i]);
|
|
}
|
|
for (int i = 126; i < 129; ++i) {
|
|
printf("%d:[0x%02x]\n", i, (uint8_t)buf[i]);
|
|
}
|
|
}
|
|
// shift by 4 bits to the left
|
|
for (int i = 0; i < bytes; ++i) {
|
|
buf[i] = ((buf[i] & 0xf) << 4) + ((buf[i + 1] >> 4) & 0xf);
|
|
}
|
|
if (totalBytesRead <= 2 * bytes) {
|
|
for (int i = 0; i < 3; ++i) {
|
|
printf("%d:[0x%02x]\n", i, (uint8_t)buf[i]);
|
|
}
|
|
for (int i = 126; i < 129; ++i) {
|
|
printf("%d:[0x%02x]\n", i, (uint8_t)buf[i]);
|
|
}
|
|
printf("\n");
|
|
}
|
|
lastByte = buf[bytes];
|
|
if (!MD5_Update(&c, buf, bytes)) {
|
|
fclose(fp);
|
|
strcpy(mess, "Unable to calculate checksum (MD5_Update)\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
if (fwrite((void *)buf, sizeof(char), bytes, flashfp) != bytes) {
|
|
LOG(logERROR, ("Could not write to flash fp file\n"));
|
|
}
|
|
// read only until a particular size (drive)
|
|
if (fsize != 0 && totalBytesRead >= fsize) {
|
|
LOG(logINFOBLUE, ("\tReached %lu bytes. Not reading more\n", totalBytesRead));
|
|
break;
|
|
}
|
|
bytes = fread(buf + 1, 1, readUnitSize, fp);
|
|
totalBytesRead += bytes;
|
|
if (totalBytesRead <= 2 * bytes) {
|
|
for (int i = 0; i < 3; ++i) {
|
|
printf("%d:[0x%02x]\n", i, (uint8_t)buf[i]);
|
|
}
|
|
for (int i = 126; i < 129; ++i) {
|
|
printf("%d:[0x%02x]\n", i, (uint8_t)buf[i]);
|
|
}
|
|
printf("\n");
|
|
}
|
|
}
|
|
LOG(logINFO, ("\tRead %lu bytes to calculate checksum\n", totalBytesRead));
|
|
fclose(fp);
|
|
fclose(flashfp);
|
|
return verifyChecksum(mess, clientChecksum, &c, "flash");
|
|
}
|
|
*/
|
|
int verifyChecksumFromFlash(char *mess, char *clientChecksum, char *fname,
|
|
ssize_t fsize) {
|
|
LOG(logINFO, ("\tVerifying FlashChecksum...\n"));
|
|
|
|
FILE *fp = fopen(fname, "r");
|
|
if (fp == NULL) {
|
|
sprintf(mess, "Unable to open %s in read mode to get checksum\n",
|
|
fname);
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
|
|
FILE *flashfp = fopen(FLASH_CONVERTED_NAME, "w");
|
|
if (flashfp == NULL) {
|
|
sprintf(mess, "Unable to open %s in write mode to write checksum\n",
|
|
FLASH_CONVERTED_NAME);
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
|
|
MD5_CTX c;
|
|
if (!MD5_Init(&c)) {
|
|
fclose(fp);
|
|
strcpy(mess, "Unable to calculate checksum (MD5_Init)\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
const int readUnitSize = 128;
|
|
int character = fgetc(fp);
|
|
int oldProgress = 0;
|
|
ssize_t totalBytesRead = 1;
|
|
|
|
while (!feof(fp)) {
|
|
int progress = (int)(((double)(totalBytesRead) / fsize) * 100);
|
|
if (oldProgress != progress) {
|
|
printf("%d%%\r", progress);
|
|
fflush(stdout);
|
|
oldProgress = progress;
|
|
}
|
|
|
|
if (!MD5_Update(&c, buf, bytes)) {
|
|
fclose(fp);
|
|
strcpy(mess, "Unable to calculate checksum (MD5_Update)\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
if (fputc(character, flashfp) != character) {
|
|
LOG(logERROR, ("Could not write to flash fp file\n"));
|
|
}
|
|
// read only until a particular size (drive)
|
|
if (fsize != 0 && totalBytesRead >= fsize) {
|
|
LOG(logINFOBLUE, ("\tReached %lu bytes. Not reading more\n", totalBytesRead));
|
|
break;
|
|
}
|
|
c = fgetc(fp);
|
|
++totalBytesRead;
|
|
}
|
|
|
|
LOG(logINFO, ("\tRead %lu bytes to calculate checksum\n", totalBytesRead));
|
|
fclose(fp);
|
|
fclose(flashfp);
|
|
return verifyChecksum(mess, clientChecksum, &c, "flash");
|
|
}
|
|
|
|
int verifyChecksum(char *mess, char *clientChecksum, MD5_CTX *c, char *msg) {
|
|
unsigned char out[MD5_DIGEST_LENGTH];
|
|
if (!MD5_Final(out, c)) {
|
|
strcpy(mess, "Unable to calculate checksum (MD5_Final)\n");
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
|
|
char checksum[512];
|
|
memset(checksum, 0, 512);
|
|
for (int i = 0; i != MD5_DIGEST_LENGTH; ++i) {
|
|
char part[16];
|
|
memset(part, 0, 16);
|
|
sprintf(part, "%02x", out[i]);
|
|
strcat(checksum, part);
|
|
}
|
|
|
|
LOG(logDEBUG1,
|
|
("\nC checksum: %s\nS checksum: %s\n", clientChecksum, checksum));
|
|
|
|
// compare checksum
|
|
if (strcmp(clientChecksum, checksum)) {
|
|
sprintf(mess,
|
|
"Checksum of %s does not match. Client "
|
|
"checksum:%s, copied checksum:%s\n",
|
|
msg, clientChecksum, checksum);
|
|
LOG(logERROR, (mess));
|
|
return FAIL;
|
|
}
|
|
LOG(logINFO, ("\tChecksum verified\n"));
|
|
return OK;
|
|
} |