2013-01-21 02:10:42 -04:00
2015-08-11 03:28:43 -03:00
# include <AP_HAL/AP_HAL.h>
2013-01-21 02:10:42 -04:00
# if CONFIG_HAL_BOARD == HAL_BOARD_PX4
# include <stdio.h>
# include <stdarg.h>
2013-02-07 00:03:54 -04:00
# include <unistd.h>
# include <stdlib.h>
# include <errno.h>
# include <apps/nsh.h>
# include <fcntl.h>
# include "UARTDriver.h"
2013-10-05 02:47:28 -03:00
# include <uORB/uORB.h>
# include <uORB/topics/safety.h>
2014-03-07 16:43:52 -04:00
# include <systemlib/board_serial.h>
2016-06-15 05:01:38 -03:00
# include <drivers/drv_gpio.h>
# include <AP_Math/AP_Math.h>
2018-06-28 19:31:41 -03:00
# include <AP_BoardConfig/AP_BoardConfig.h>
2018-07-25 13:50:02 -03:00
# include <drivers/drv_tone_alarm.h>
2013-02-07 00:03:54 -04:00
extern const AP_HAL : : HAL & hal ;
2013-01-21 02:10:42 -04:00
# include "Util.h"
using namespace PX4 ;
2013-02-07 00:03:54 -04:00
extern bool _px4_thread_should_exit ;
2013-10-05 02:47:28 -03:00
/*
constructor
*/
2015-02-11 05:56:47 -04:00
PX4Util : : PX4Util ( void ) : Util ( )
2013-10-05 02:47:28 -03:00
{
_safety_handle = orb_subscribe ( ORB_ID ( safety ) ) ;
}
2013-02-07 00:03:54 -04:00
/*
start an instance of nsh
*/
bool PX4Util : : run_debug_shell ( AP_HAL : : BetterStream * stream )
{
2015-06-17 04:04:51 -03:00
PX4UARTDriver * uart = ( PX4UARTDriver * ) stream ;
int fd ;
// trigger exit in the other threads. This stops use of the
// various driver handles, and especially the px4io handle,
// which otherwise would cause a crash if px4io is stopped in
// the shell
_px4_thread_should_exit = true ;
// take control of stream fd
fd = uart - > _get_fd ( ) ;
// mark it blocking (nsh expects a blocking fd)
unsigned v ;
v = fcntl ( fd , F_GETFL , 0 ) ;
fcntl ( fd , F_SETFL , v & ~ O_NONBLOCK ) ;
// setup the UART on stdin/stdout/stderr
close ( 0 ) ;
close ( 1 ) ;
close ( 2 ) ;
dup2 ( fd , 0 ) ;
dup2 ( fd , 1 ) ;
dup2 ( fd , 2 ) ;
2016-10-30 02:24:21 -03:00
nsh_consolemain ( 0 , nullptr ) ;
2015-06-17 04:04:51 -03:00
// this shouldn't happen
hal . console - > printf ( " shell exited \n " ) ;
return true ;
2013-02-07 00:03:54 -04:00
}
2013-10-05 02:47:28 -03:00
/*
return state of safety switch
*/
enum PX4Util : : safety_state PX4Util : : safety_switch_state ( void )
{
2017-03-20 15:58:36 -03:00
# if !HAL_HAVE_SAFETY_SWITCH
2017-02-09 23:18:13 -04:00
return AP_HAL : : Util : : SAFETY_NONE ;
2017-03-20 15:58:36 -03:00
# endif
2013-10-05 02:47:28 -03:00
if ( _safety_handle = = - 1 ) {
_safety_handle = orb_subscribe ( ORB_ID ( safety ) ) ;
}
if ( _safety_handle = = - 1 ) {
return AP_HAL : : Util : : SAFETY_NONE ;
}
struct safety_s safety ;
if ( orb_copy ( ORB_ID ( safety ) , _safety_handle , & safety ) ! = OK ) {
return AP_HAL : : Util : : SAFETY_NONE ;
}
if ( ! safety . safety_switch_available ) {
return AP_HAL : : Util : : SAFETY_NONE ;
}
if ( safety . safety_off ) {
return AP_HAL : : Util : : SAFETY_ARMED ;
}
return AP_HAL : : Util : : SAFETY_DISARMED ;
}
2013-11-25 20:57:59 -04:00
/*
display PX4 system identifer - board type and serial number
*/
bool PX4Util : : get_system_id ( char buf [ 40 ] )
{
uint8_t serialid [ 12 ] ;
memset ( serialid , 0 , sizeof ( serialid ) ) ;
2014-03-07 16:43:52 -04:00
get_board_serial ( serialid ) ;
2016-07-18 13:25:01 -03:00
# if defined(CONFIG_ARCH_BOARD_PX4FMU_V1)
2013-11-25 20:57:59 -04:00
const char * board_type = " PX4v1 " ;
2017-04-02 23:49:28 -03:00
# elif defined(CONFIG_ARCH_BOARD_PX4FMU_V3)
const char * board_type = " PX4v3 " ;
2016-07-18 13:25:01 -03:00
# elif defined(CONFIG_ARCH_BOARD_PX4FMU_V2)
2013-11-25 20:57:59 -04:00
const char * board_type = " PX4v2 " ;
2016-07-18 13:25:01 -03:00
# elif defined(CONFIG_ARCH_BOARD_PX4FMU_V4)
2016-04-14 08:53:07 -03:00
const char * board_type = " PX4v4 " ;
2017-07-15 02:12:39 -03:00
# elif defined(CONFIG_ARCH_BOARD_PX4FMU_V4PRO)
const char * board_type = " PX4v4PRO " ;
2017-01-27 05:21:12 -04:00
# elif defined(CONFIG_ARCH_BOARD_AEROFC_V1)
const char * board_type = " AEROFCv1 " ;
2016-04-14 08:53:07 -03:00
# else
const char * board_type = " PX4v? " ;
2013-11-25 20:57:59 -04:00
# endif
// this format is chosen to match the human_readable_serial()
// function in auth.c
snprintf ( buf , 40 , " %s %02X%02X%02X%02X %02X%02X%02X%02X %02X%02X%02X%02X " ,
board_type ,
( unsigned ) serialid [ 0 ] , ( unsigned ) serialid [ 1 ] , ( unsigned ) serialid [ 2 ] , ( unsigned ) serialid [ 3 ] ,
( unsigned ) serialid [ 4 ] , ( unsigned ) serialid [ 5 ] , ( unsigned ) serialid [ 6 ] , ( unsigned ) serialid [ 7 ] ,
2018-07-10 01:41:16 -03:00
( unsigned ) serialid [ 8 ] , ( unsigned ) serialid [ 9 ] , ( unsigned ) serialid [ 10 ] , ( unsigned ) serialid [ 11 ] ) ;
buf [ 39 ] = 0 ;
2013-11-25 20:57:59 -04:00
return true ;
}
2013-12-28 01:01:28 -04:00
/**
how much free memory do we have in bytes .
*/
2015-11-05 00:41:25 -04:00
uint32_t PX4Util : : available_memory ( void )
2013-12-28 01:01:28 -04:00
{
2015-11-05 00:41:25 -04:00
return mallinfo ( ) . fordblks ;
2013-12-28 01:01:28 -04:00
}
2015-10-20 02:42:18 -03:00
/*
AP_HAL wrapper around PX4 perf counters
*/
PX4Util : : perf_counter_t PX4Util : : perf_alloc ( PX4Util : : perf_counter_type t , const char * name )
{
: : perf_counter_type px4_t ;
switch ( t ) {
case PX4Util : : PC_COUNT :
px4_t = : : PC_COUNT ;
break ;
case PX4Util : : PC_ELAPSED :
px4_t = : : PC_ELAPSED ;
break ;
case PX4Util : : PC_INTERVAL :
px4_t = : : PC_INTERVAL ;
break ;
default :
2016-10-30 02:24:21 -03:00
return nullptr ;
2015-10-20 02:42:18 -03:00
}
return ( perf_counter_t ) : : perf_alloc ( px4_t , name ) ;
}
void PX4Util : : perf_begin ( perf_counter_t h )
{
: : perf_begin ( ( : : perf_counter_t ) h ) ;
}
void PX4Util : : perf_end ( perf_counter_t h )
{
: : perf_end ( ( : : perf_counter_t ) h ) ;
}
void PX4Util : : perf_count ( perf_counter_t h )
{
: : perf_count ( ( : : perf_counter_t ) h ) ;
}
2016-06-15 05:01:38 -03:00
void PX4Util : : set_imu_temp ( float current )
{
if ( ! _heater . target | | * _heater . target = = - 1 ) {
return ;
}
// average over temperatures to remove noise
_heater . count + + ;
_heater . sum + = current ;
// update once a second
uint32_t now = AP_HAL : : millis ( ) ;
if ( now - _heater . last_update_ms < 1000 ) {
return ;
}
_heater . last_update_ms = now ;
current = _heater . sum / _heater . count ;
_heater . sum = 0 ;
_heater . count = 0 ;
// experimentally tweaked for Pixhawk2
const float kI = 0.3f ;
const float kP = 200.0f ;
2016-09-04 23:59:50 -03:00
float target = ( float ) ( * _heater . target ) ;
// limit to 65 degrees to prevent damage
target = constrain_float ( target , 0 , 65 ) ;
2016-06-15 05:01:38 -03:00
2016-09-04 23:59:50 -03:00
float err = target - current ;
2016-06-15 05:01:38 -03:00
_heater . integrator + = kI * err ;
_heater . integrator = constrain_float ( _heater . integrator , 0 , 70 ) ;
float output = constrain_float ( kP * err + _heater . integrator , 0 , 100 ) ;
// hal.console->printf("integrator %.1f out=%.1f temp=%.2f err=%.2f\n", _heater.integrator, output, current, err);
if ( _heater . fd = = - 1 ) {
_heater . fd = open ( " /dev/px4io " , O_RDWR ) ;
}
if ( _heater . fd ! = - 1 ) {
ioctl ( _heater . fd , GPIO_SET_HEATER_DUTY_CYCLE , ( unsigned ) output ) ;
}
}
void PX4Util : : set_imu_target_temp ( int8_t * target )
{
_heater . target = target ;
}
2016-11-23 05:30:48 -04:00
extern " C " {
extern void * fat_dma_alloc ( size_t ) ;
extern void fat_dma_free ( void * , size_t ) ;
}
/*
allocate DMA - capable memory if possible . Otherwise return normal
memory .
*/
2018-01-09 17:42:54 -04:00
void * PX4Util : : malloc_type ( size_t size , AP_HAL : : Util : : Memory_Type mem_type )
2016-11-23 05:30:48 -04:00
{
2017-03-10 17:01:40 -04:00
# if !defined(CONFIG_ARCH_BOARD_PX4FMU_V1)
2018-01-09 17:42:54 -04:00
if ( mem_type = = AP_HAL : : Util : : MEM_DMA_SAFE ) {
return fat_dma_alloc ( size ) ;
} else {
2018-01-16 17:02:46 -04:00
return calloc ( 1 , size ) ;
2018-01-09 17:42:54 -04:00
}
2016-11-26 16:45:54 -04:00
# else
2018-01-16 17:02:46 -04:00
return calloc ( 1 , size ) ;
2016-11-26 16:45:54 -04:00
# endif
2016-11-23 05:30:48 -04:00
}
2018-01-09 17:42:54 -04:00
void PX4Util : : free_type ( void * ptr , size_t size , AP_HAL : : Util : : Memory_Type mem_type )
2016-11-23 05:30:48 -04:00
{
2017-03-10 17:01:40 -04:00
# if !defined(CONFIG_ARCH_BOARD_PX4FMU_V1)
2018-01-09 17:42:54 -04:00
if ( mem_type = = AP_HAL : : Util : : MEM_DMA_SAFE ) {
return fat_dma_free ( ptr , size ) ;
} else {
return free ( ptr ) ;
}
2016-11-26 16:45:54 -04:00
# else
return free ( ptr ) ;
# endif
2016-11-23 05:30:48 -04:00
}
2018-06-28 19:31:41 -03:00
extern " C " {
int bl_update_main ( int argc , char * argv [ ] ) ;
} ;
bool PX4Util : : flash_bootloader ( )
{
2018-07-06 06:59:14 -03:00
# if !defined(CONFIG_ARCH_BOARD_AEROFC_V1)
2018-06-28 19:31:41 -03:00
if ( AP_BoardConfig : : px4_start_driver ( bl_update_main , " bl_update " , " /etc/bootloader/fmu_bl.bin " ) ) {
hal . console - > printf ( " updated bootloader \n " ) ;
return true ;
}
2018-07-06 06:59:14 -03:00
# endif
2018-06-28 19:31:41 -03:00
return false ;
}
2018-07-25 13:50:02 -03:00
bool PX4Util : : toneAlarm_init ( ) {
_tonealarm_fd = open ( TONEALARM0_DEVICE_PATH , O_WRONLY ) ;
if ( _tonealarm_fd = = - 1 ) {
hal . console - > printf ( " ToneAlarm_PX4: Unable to open " TONEALARM0_DEVICE_PATH ) ;
return false ;
}
return true ;
}
void PX4Util : : toneAlarm_set_buzzer_tone ( float frequency , float volume ) {
if ( is_zero ( frequency ) | | is_zero ( volume ) ) {
write ( _tonealarm_fd , " MFP " , 4 ) ;
} else {
unsigned note = constrain_float ( roundf ( 17.3123404906676f * logf ( frequency ) - 71.3763165622959f ) , 4 , 87 ) ; // constraint based on comment in PX4 tonealarm driver - "note value in the range C1 to B7"
char tune [ 20 ] ;
snprintf ( tune , sizeof ( tune ) , " MFMLT32L1N%u " , note ) ;
tune [ sizeof ( tune ) - 1 ] = ' \0 ' ;
write ( _tonealarm_fd , tune , strlen ( tune ) + 1 ) ;
}
}
2018-06-28 19:31:41 -03:00
2013-01-21 02:10:42 -04:00
# endif // CONFIG_HAL_BOARD == HAL_BOARD_PX4