You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
288 lines
6.6 KiB
288 lines
6.6 KiB
#include <errno.h> |
|
#include <fcntl.h> |
|
#include <stdarg.h> |
|
#include <stdio.h> |
|
#include <stdlib.h> |
|
#include <sys/stat.h> |
|
#include <time.h> |
|
#include <unistd.h> |
|
|
|
#include <AP_HAL/AP_HAL.h> |
|
|
|
#include "Heat_Pwm.h" |
|
#include "ToneAlarm_Disco.h" |
|
#include "Util.h" |
|
|
|
using namespace Linux; |
|
|
|
extern const AP_HAL::HAL& hal; |
|
|
|
#if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_LINUX_DISCO |
|
ToneAlarm_Disco Util::_toneAlarm; |
|
#else |
|
ToneAlarm Util::_toneAlarm; |
|
#endif |
|
|
|
void Util::init(int argc, char * const *argv) { |
|
saved_argc = argc; |
|
saved_argv = argv; |
|
|
|
#ifdef HAL_UTILS_HEAT |
|
#if HAL_UTILS_HEAT == HAL_LINUX_HEAT_PWM |
|
_heat = new Linux::HeatPwm(HAL_LINUX_HEAT_PWM_NUM, |
|
HAL_LINUX_HEAT_KP, |
|
HAL_LINUX_HEAT_KI, |
|
HAL_LINUX_HEAT_PERIOD_NS); |
|
#else |
|
#error Unrecognized Heat |
|
#endif // #if |
|
#else |
|
_heat = new Linux::Heat(); |
|
#endif // #ifdef |
|
} |
|
|
|
// set current IMU temperatue in degrees C |
|
void Util::set_imu_temp(float current) |
|
{ |
|
_heat->set_imu_temp(current); |
|
} |
|
|
|
// set target IMU temperatue in degrees C |
|
void Util::set_imu_target_temp(int8_t *target) |
|
{ |
|
_heat->set_imu_target_temp(target); |
|
} |
|
|
|
/** |
|
return commandline arguments, if available |
|
*/ |
|
void Util::commandline_arguments(uint8_t &argc, char * const *&argv) |
|
{ |
|
argc = saved_argc; |
|
argv = saved_argv; |
|
} |
|
|
|
void Util::set_hw_rtc(uint64_t time_utc_usec) |
|
{ |
|
#if CONFIG_HAL_BOARD_SUBTYPE != HAL_BOARD_SUBTYPE_LINUX_NONE |
|
// call superclass method to set time. We've guarded this so we |
|
// don't reset the HW clock time on people's laptops. |
|
AP_HAL::Util::set_hw_rtc(time_utc_usec); |
|
#endif |
|
} |
|
|
|
bool Util::is_chardev_node(const char *path) |
|
{ |
|
struct stat st; |
|
|
|
if (!path || lstat(path, &st) < 0) { |
|
return false; |
|
} |
|
|
|
return S_ISCHR(st.st_mode); |
|
} |
|
|
|
/* |
|
always report 256k of free memory. Using mallinfo() isn't useful as |
|
it only reported the current heap, which auto-expands. What we're |
|
trying to do here is ensure that code which checks for free memory |
|
before allocating objects does allow the allocation |
|
*/ |
|
uint32_t Util::available_memory(void) |
|
{ |
|
return 256*1024; |
|
} |
|
|
|
#ifndef HAL_LINUX_DEFAULT_SYSTEM_ID |
|
#define HAL_LINUX_DEFAULT_SYSTEM_ID "linux-unknown" |
|
#endif |
|
|
|
/* |
|
get a (hopefully unique) machine ID |
|
*/ |
|
bool Util::get_system_id_unformatted(uint8_t buf[], uint8_t &len) |
|
{ |
|
char *cbuf = (char *)buf; |
|
|
|
// try first to use machine-id file. Most systems will have this |
|
const char *paths[] = { "/etc/machine-id", "/var/lib/dbus/machine-id" }; |
|
for (uint8_t i=0; i<ARRAY_SIZE(paths); i++) { |
|
int fd = open(paths[i], O_RDONLY); |
|
if (fd == -1) { |
|
continue; |
|
} |
|
ssize_t ret = read(fd, buf, len); |
|
close(fd); |
|
if (ret <= 0) { |
|
continue; |
|
} |
|
len = ret; |
|
char *p = strchr(cbuf, '\n'); |
|
if (p) { |
|
*p = 0; |
|
} |
|
len = strnlen(cbuf, len); |
|
return true; |
|
} |
|
|
|
// fallback to hostname |
|
if (gethostname(cbuf, len) != 0) { |
|
// use a default name so this always succeeds. Without it we can't |
|
// implement some features (such as UAVCAN) |
|
strncpy(cbuf, HAL_LINUX_DEFAULT_SYSTEM_ID, len); |
|
} |
|
len = strnlen(cbuf, len); |
|
return true; |
|
} |
|
|
|
/* |
|
as get_system_id_unformatted will already be ascii, we use the same |
|
ID here |
|
*/ |
|
bool Util::get_system_id(char buf[40]) |
|
{ |
|
uint8_t len = 40; |
|
return get_system_id_unformatted((uint8_t *)buf, len); |
|
} |
|
|
|
|
|
int Util::write_file(const char *path, const char *fmt, ...) |
|
{ |
|
errno = 0; |
|
|
|
int fd = open(path, O_WRONLY | O_CLOEXEC); |
|
if (fd == -1) { |
|
return -errno; |
|
} |
|
|
|
va_list args; |
|
va_start(args, fmt); |
|
|
|
int ret = vdprintf(fd, fmt, args); |
|
int errno_bkp = errno; |
|
close(fd); |
|
|
|
va_end(args); |
|
|
|
if (ret < 1) { |
|
return -errno_bkp; |
|
} |
|
|
|
return ret; |
|
} |
|
|
|
int Util::read_file(const char *path, const char *fmt, ...) |
|
{ |
|
errno = 0; |
|
|
|
FILE *file = fopen(path, "re"); |
|
if (!file) { |
|
return -errno; |
|
} |
|
|
|
va_list args; |
|
va_start(args, fmt); |
|
|
|
int ret = vfscanf(file, fmt, args); |
|
int errno_bkp = errno; |
|
fclose(file); |
|
|
|
va_end(args); |
|
|
|
if (ret < 1) { |
|
return -errno_bkp; |
|
} |
|
|
|
return ret; |
|
} |
|
|
|
const char *Linux::Util::_hw_names[UTIL_NUM_HARDWARES] = { |
|
[UTIL_HARDWARE_RPI1] = "BCM2708", |
|
[UTIL_HARDWARE_RPI2] = "BCM2709", |
|
[UTIL_HARDWARE_BEBOP] = "Mykonos3 board", |
|
[UTIL_HARDWARE_BEBOP2] = "Milos board", |
|
[UTIL_HARDWARE_DISCO] = "Evinrude board", |
|
}; |
|
|
|
#define MAX_SIZE_LINE 50 |
|
int Util::get_hw_arm32() |
|
{ |
|
char buffer[MAX_SIZE_LINE] = { 0 }; |
|
FILE *f = fopen("/proc/cpuinfo", "r"); |
|
if (f == nullptr) { |
|
return -errno; |
|
} |
|
|
|
while (fgets(buffer, MAX_SIZE_LINE, f) != nullptr) { |
|
if (strstr(buffer, "Hardware") == nullptr) { |
|
continue; |
|
} |
|
for (uint8_t i = 0; i < UTIL_NUM_HARDWARES; i++) { |
|
if (strstr(buffer, _hw_names[i]) == nullptr) { |
|
continue; |
|
} |
|
fclose(f); |
|
return i; |
|
} |
|
} |
|
|
|
fclose(f); |
|
return -ENOENT; |
|
} |
|
|
|
#ifdef ENABLE_HEAP |
|
void *Util::allocate_heap_memory(size_t size) |
|
{ |
|
struct heap *new_heap = (struct heap*)malloc(sizeof(struct heap)); |
|
if (new_heap != nullptr) { |
|
new_heap->max_heap_size = size; |
|
new_heap->current_heap_usage = 0; |
|
} |
|
return (void *)new_heap; |
|
} |
|
|
|
void *Util::heap_realloc(void *h, void *ptr, size_t new_size) |
|
{ |
|
if (h == nullptr) { |
|
return nullptr; |
|
} |
|
|
|
struct heap *heapp = (struct heap*)h; |
|
|
|
// extract appropriate headers |
|
size_t old_size = 0; |
|
heap_allocation_header *old_header = nullptr; |
|
if (ptr != nullptr) { |
|
old_header = ((heap_allocation_header *)ptr) - 1; |
|
old_size = old_header->allocation_size; |
|
} |
|
|
|
if ((heapp->current_heap_usage + new_size - old_size) > heapp->max_heap_size) { |
|
// fail the allocation as we don't have the memory. Note that we don't simulate fragmentation |
|
return nullptr; |
|
} |
|
|
|
heapp->current_heap_usage -= old_size; |
|
if (new_size == 0) { |
|
free(old_header); |
|
return nullptr; |
|
} |
|
|
|
heap_allocation_header *new_header = (heap_allocation_header *)malloc(new_size + sizeof(heap_allocation_header)); |
|
if (new_header == nullptr) { |
|
// total failure to allocate, this is very surprising in SITL |
|
return nullptr; |
|
} |
|
heapp->current_heap_usage += new_size; |
|
new_header->allocation_size = new_size; |
|
void *new_mem = new_header + 1; |
|
|
|
if (ptr == nullptr) { |
|
return new_mem; |
|
} |
|
memcpy(new_mem, ptr, old_size > new_size ? new_size : old_size); |
|
free(old_header); |
|
return new_mem; |
|
} |
|
|
|
#endif // ENABLE_HEAP
|
|
|