Skip to content
GitLab
Explore
Sign in
Primary navigation
Search or go to…
Project
F
Firmware
Manage
Activity
Members
Labels
Plan
Issues
Issue boards
Milestones
Wiki
Code
Merge requests
Repository
Branches
Commits
Tags
Repository graph
Compare revisions
Snippets
Build
Pipelines
Jobs
Pipeline schedules
Artifacts
Deploy
Releases
Package Registry
Model registry
Operate
Environments
Terraform modules
Monitor
Incidents
Analyze
Value stream analytics
Contributor analytics
CI/CD analytics
Repository analytics
Model experiments
Help
Help
Support
GitLab documentation
Compare GitLab plans
Community forum
Contribute to GitLab
Provide feedback
Terms and privacy
Keyboard shortcuts
?
Snippets
Groups
Projects
Show more breadcrumbs
Alberto Ruiz Garcia
Firmware
Commits
250c5619
Commit
250c5619
authored
6 years ago
by
David Sidrane
Committed by
Lorenz Meier
6 years ago
Browse files
Options
Downloads
Patches
Plain Diff
posix simulated tone_alarm: Use tune_control
Refactor to use the tune_control interface
parent
1fb70460
No related branches found
Branches containing commit
No related tags found
Tags containing commit
No related merge requests found
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
src/platforms/posix/drivers/tonealrmsim/tone_alarm.cpp
+163
-679
163 additions, 679 deletions
src/platforms/posix/drivers/tonealrmsim/tone_alarm.cpp
with
163 additions
and
679 deletions
src/platforms/posix/drivers/tonealrmsim/tone_alarm.cpp
+
163
−
679
View file @
250c5619
/****************************************************************************
*
* Copyright (c) 2013, 2017 PX4 Development Team. All rights reserved.
* Copyright (c) 2013, 2017
-2018
PX4 Development Team. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
...
...
@@ -31,59 +31,10 @@
*
****************************************************************************/
/**
* Driver for the PX4 audio alarm port, /dev/tone_alarm.
*
* The tone_alarm driver supports a set of predefined "alarm"
* tunes and one user-supplied tune.
*
* The TONE_SET_ALARM ioctl can be used to select a predefined
* alarm tune, from 1 - <TBD>. Selecting tune zero silences
* the alarm.
*
* Tunes follow the syntax of the Microsoft GWBasic/QBasic PLAY
* statement, with some exceptions and extensions.
*
* From Wikibooks:
*
* PLAY "[string expression]"
*
* Used to play notes and a score ... The tones are indicated by letters A through G.
* Accidentals are indicated with a "+" or "#" (for sharp) or "-" (for flat)
* immediately after the note letter. See this example:
*
* PLAY "C C# C C#"
*
* Whitespaces are ignored inside the string expression. There are also codes that
* set the duration, octave and tempo. They are all case-insensitive. PLAY executes
* the commands or notes the order in which they appear in the string. Any indicators
* that change the properties are effective for the notes following that indicator.
*
* Ln Sets the duration (length) of the notes. The variable n does not indicate an actual duration
* amount but rather a note type; L1 - whole note, L2 - half note, L4 - quarter note, etc.
* (L8, L16, L32, L64, ...). By default, n = 4.
* For triplets and quintets, use L3, L6, L12, ... and L5, L10, L20, ... series respectively.
* The shorthand notation of length is also provided for a note. For example, "L4 CDE L8 FG L4 AB"
* can be shortened to "L4 CDE F8G8 AB". F and G play as eighth notes while others play as quarter notes.
* On Sets the current octave. Valid values for n are 0 through 6. An octave begins with C and ends with B.
* Remember that C- is equivalent to B.
* < > Changes the current octave respectively down or up one level.
* Nn Plays a specified note in the seven-octave range. Valid values are from 0 to 84. (0 is a pause.)
* Cannot use with sharp and flat. Cannot use with the shorthand notation neither.
* MN Stand for Music Normal. Note duration is 7/8ths of the length indicated by Ln. It is the default mode.
* ML Stand for Music Legato. Note duration is full length of that indicated by Ln.
* MS Stand for Music Staccato. Note duration is 3/4ths of the length indicated by Ln.
* Pn Causes a silence (pause) for the length of note indicated (same as Ln).
* Tn Sets the number of "L4"s per minute (tempo). Valid values are from 32 to 255. The default value is T120.
* . When placed after a note, it causes the duration of the note to be 3/2 of the set duration.
* This is how to get "dotted" notes. "L4 C#." would play C sharp as a dotted quarter note.
* It can be used for a pause as well.
*
* Extensions/variations:
*
* MB MF The MF command causes the tune to play once and then stop. The MB command causes the
* tune to repeat when it ends.
*
/*
* Simulated Low Level Driver for the PX4 audio alarm port. Subscribes to
* tune_control and plays notes on this architecture specific
* timer HW
*/
#include
<px4_config.h>
...
...
@@ -108,102 +59,77 @@
#include
<drivers/drv_hrt.h>
#include
<systemlib/err.h>
#include
<circuit_breaker/circuit_breaker.h>
#include
<lib/tunes/tunes.h>
#include
<uORB/uORB.h>
#include
<uORB/topics/tune_control.h>
#include
"VirtDevObj.hpp"
using
namespace
DriverFramework
;
#if !defined(UNUSED)
# define UNUSED(a) ((void)(a))
#endif
#define CBRK_BUZZER_KEY 782097
class
ToneAlarm
:
public
VirtDevObj
{
public:
ToneAlarm
();
~
ToneAlarm
()
=
default
;
virtual
int
devIOCTL
(
unsigned
long
cmd
,
unsigned
long
arg
);
virtual
ssize_t
devWrite
(
const
void
*
buffer
,
size_t
len
);
inline
const
char
*
name
(
int
tune
)
{
return
_tune_names
[
tune
];
}
private
:
static
const
unsigned
_tune_max
=
1024
*
8
;
// be reasonable about user tunes
const
char
*
_default_tunes
[
TONE_NUMBER_OF_TUNES
];
const
char
*
_tune_names
[
TONE_NUMBER_OF_TUNES
];
static
const
uint8_t
_note_tab
[];
~
ToneAlarm
();
unsigned
_default_tune_number
;
// number of currently playing default tune (0 for none)
virtual
int
init
();
void
status
();
const
char
*
_user_tune
;
enum
{
CBRK_OFF
=
0
,
CBRK_ON
,
CBRK_UNINIT
};
const
char
*
_tune
;
// current tune string
const
char
*
_next
;
// next note in the string
private
:
volatile
bool
_running
;
volatile
bool
_should_run
;
bool
_play_tone
;
unsigned
_tempo
;
unsigned
_note_length
;
enum
{
MODE_NORMAL
,
MODE_LEGATO
,
MODE_STACCATO
}
_note_mode
;
unsigned
_octave
;
unsigned
_silence_length
;
// if nonzero, silence before next note
bool
_repeat
;
// if true, tune restarts at end
Tunes
_tunes
;
hrt_call
_note_call
;
// HRT callout for note completion
// Convert a note value in the range C1 to B7 into a divisor for
// the configured timer's clock.
//
unsigned
note_to_divisor
(
unsigned
note
);
unsigned
_silence_length
;
// if nonzero, silence before next note
// Calculate the duration in microseconds of play and silence for a
// note given the current tempo, length and mode and the number of
// dots following in the play string.
//
unsigned
note_duration
(
unsigned
&
silence
,
unsigned
note_length
,
unsigned
dots
);
int
_cbrk
;
///< if true, no audio output
int
_tune_control_sub
;
// Calculate the duration in microseconds of a rest corresponding to
// a given note length.
tune_control_s
_tune
;
// Convert a frequency value into a divisor for the configured timer's clock.
//
unsigned
rest_duration
(
unsigned
re
st_length
,
unsigned
dots
);
unsigned
frequency_to_divisor
(
unsigned
f
re
quency
);
// Start playing the note
//
void
start_note
(
unsigned
note
);
void
start_note
(
unsigned
frequency
);
// Stop playing the current note and make the player 'safe'
//
void
stop_note
();
// Start playing the tune
//
void
start_tune
(
const
char
*
tune
);
void
stop_note
();
// Parse the next note out of the string and play it
//
void
next_note
();
// Find the next character in the string, discard any whitespace and
// return the canonical (uppercase) version.
//
int
next_char
();
// Extract a number from the string, consuming all the digit characters.
//
unsigned
next_number
();
// Consume dot characters from the string, returning the number consumed.
//
unsigned
next_dots
();
void
next_note
();
// hrt_call trampoline for next_note
//
static
void
next_trampoline
(
void
*
arg
);
static
void
next_trampoline
(
void
*
arg
);
// Unused
virtual
void
_measure
()
{}
};
// semitone offsets from C for the characters 'A'-'G'
const
uint8_t
ToneAlarm
::
_note_tab
[]
=
{
9
,
11
,
0
,
2
,
4
,
5
,
7
};
/*
* Driver 'main' command.
*/
...
...
@@ -212,128 +138,75 @@ extern "C" __EXPORT int tone_alarm_main(int argc, char *argv[]);
ToneAlarm
::
ToneAlarm
()
:
VirtDevObj
(
"tone_alarm"
,
TONEALARM0_DEVICE_PATH
,
nullptr
,
0
),
_default_tune_number
(
0
),
_user_tune
(
nullptr
),
_tune
(
nullptr
),
_next
(
nullptr
),
_note_call
{}
_running
(
false
),
_should_run
(
true
),
_play_tone
(
false
),
_tunes
(),
_silence_length
(
0
),
_cbrk
(
CBRK_UNINIT
),
_tune_control_sub
(
-
1
)
{
_default_tunes
[
TONE_STARTUP_TUNE
]
=
"MFT240L8 O4aO5dc O4aO5dc O4aO5dc L16dcdcdcdc"
;
// startup tune
_default_tunes
[
TONE_ERROR_TUNE
]
=
"MBT200a8a8a8PaaaP"
;
// ERROR tone
_default_tunes
[
TONE_NOTIFY_POSITIVE_TUNE
]
=
"MFT200e8a8a"
;
// Notify Positive tone
_default_tunes
[
TONE_NOTIFY_NEUTRAL_TUNE
]
=
"MFT200e8e"
;
// Notify Neutral tone
_default_tunes
[
TONE_NOTIFY_NEGATIVE_TUNE
]
=
"MFT200e8c8e8c8e8c8"
;
// Notify Negative tone
_default_tunes
[
TONE_ARMING_WARNING_TUNE
]
=
"MNT75L1O2G"
;
//arming warning
_default_tunes
[
TONE_BATTERY_WARNING_SLOW_TUNE
]
=
"MBNT100a8"
;
//battery warning slow
_default_tunes
[
TONE_BATTERY_WARNING_FAST_TUNE
]
=
"MBNT255a8a8a8a8a8a8a8a8a8a8a8a8a8a8a8a8"
;
//battery warning fast
_default_tunes
[
TONE_GPS_WARNING_TUNE
]
=
"MFT255L4AAAL1F#"
;
//gps warning slow
_default_tunes
[
TONE_ARMING_FAILURE_TUNE
]
=
"MFT255L4<<<BAP"
;
_default_tunes
[
TONE_PARACHUTE_RELEASE_TUNE
]
=
"MFT255L16agagagag"
;
// parachute release
_default_tunes
[
TONE_EKF_WARNING_TUNE
]
=
"MFT255L8ddd#d#eeff"
;
// ekf warning
_default_tunes
[
TONE_BARO_WARNING_TUNE
]
=
"MFT255L4gf#fed#d"
;
// baro warning
_default_tunes
[
TONE_SINGLE_BEEP_TUNE
]
=
"MFT100a8"
;
// single beep
_tune_names
[
TONE_STARTUP_TUNE
]
=
"startup"
;
// startup tune
_tune_names
[
TONE_ERROR_TUNE
]
=
"error"
;
// ERROR tone
_tune_names
[
TONE_NOTIFY_POSITIVE_TUNE
]
=
"positive"
;
// Notify Positive tone
_tune_names
[
TONE_NOTIFY_NEUTRAL_TUNE
]
=
"neutral"
;
// Notify Neutral tone
_tune_names
[
TONE_NOTIFY_NEGATIVE_TUNE
]
=
"negative"
;
// Notify Negative tone
_tune_names
[
TONE_ARMING_WARNING_TUNE
]
=
"arming"
;
// arming warning
_tune_names
[
TONE_BATTERY_WARNING_SLOW_TUNE
]
=
"slow_bat"
;
// battery warning slow
_tune_names
[
TONE_BATTERY_WARNING_FAST_TUNE
]
=
"fast_bat"
;
// battery warning fast
_tune_names
[
TONE_GPS_WARNING_TUNE
]
=
"gps_warning"
;
// gps warning
_tune_names
[
TONE_ARMING_FAILURE_TUNE
]
=
"arming_failure"
;
//fail to arm
_tune_names
[
TONE_PARACHUTE_RELEASE_TUNE
]
=
"parachute_release"
;
// parachute release
_tune_names
[
TONE_EKF_WARNING_TUNE
]
=
"ekf_warning"
;
// ekf warning
_tune_names
[
TONE_BARO_WARNING_TUNE
]
=
"baro_warning"
;
// baro warning
_tune_names
[
TONE_SINGLE_BEEP_TUNE
]
=
"beep"
;
// single beep
_tune_names
[
TONE_HOME_SET
]
=
"home_set"
;
}
unsigned
ToneAlarm
::
note_to_divisor
(
unsigned
note
)
ToneAlarm
::~
ToneAlarm
()
{
const
int
TONE_ALARM_CLOCK
=
120000000ul
/
4
;
_should_run
=
false
;
int
counter
=
0
;
// compute the frequency first (Hz)
float
freq
=
880.0
f
*
expf
(
logf
(
2.0
f
)
*
((
int
)
note
-
46
)
/
12.0
f
);
float
period
=
0.5
f
/
freq
;
// and the divisor, rounded to the nearest integer
unsigned
divisor
=
(
period
*
TONE_ALARM_CLOCK
)
+
0.5
f
;
return
divisor
;
while
(
_running
&&
++
counter
<
10
)
{
usleep
(
100000
);
}
}
unsigned
ToneAlarm
::
note_duration
(
unsigned
&
silence
,
unsigned
note_length
,
unsigned
dots
)
int
ToneAlarm
::
init
()
{
unsigned
whole_note_period
=
(
60
*
1000000
*
4
)
/
_tempo
;
if
(
note_length
==
0
)
{
note_length
=
1
;
}
unsigned
note_period
=
whole_note_period
/
note_length
;
switch
(
_note_mode
)
{
case
MODE_NORMAL
:
silence
=
note_period
/
8
;
break
;
case
MODE_STACCATO
:
silence
=
note_period
/
4
;
break
;
default:
case
MODE_LEGATO
:
silence
=
0
;
break
;
}
note_period
-=
silence
;
int
ret
;
unsigned
dot_extension
=
note_period
/
2
;
ret
=
VirtDevObj
::
init
()
;
while
(
dots
--
)
{
note_period
+=
dot_extension
;
dot_extension
/=
2
;
if
(
ret
!=
OK
)
{
return
ret
;
}
return
note_period
;
_note_call
=
{};
hrt_call_after
(
&
_note_call
,
(
hrt_abstime
)
TUNE_MAX_UPDATE_INTERVAL_US
,
(
hrt_callout
)
next_trampoline
,
this
);
_running
=
true
;
return
OK
;
}
unsigned
ToneAlarm
::
rest_duration
(
unsigned
rest_length
,
unsigned
dots
)
void
ToneAlarm
::
status
()
{
unsigned
whole_note_period
=
(
60
*
1000000
*
4
)
/
_tempo
;
if
(
_running
)
{
PX4_INFO
(
"running"
);
if
(
rest_length
==
0
)
{
rest_length
=
1
;
}
else
{
PX4_INFO
(
"stopped"
)
;
}
}
unsigned
rest_period
=
whole_note_period
/
rest_length
;
unsigned
ToneAlarm
::
frequency_to_divisor
(
unsigned
frequency
)
{
const
int
TONE_ALARM_CLOCK
=
120000000ul
/
4
;
unsigned
dot_extension
=
rest_period
/
2
;
float
period
=
0.5
f
/
frequency
;
while
(
dots
--
)
{
rest_period
+=
dot_extension
;
dot_extension
/=
2
;
}
// and the divisor, rounded to the nearest integer
unsigned
divisor
=
(
period
*
TONE_ALARM_CLOCK
)
+
0.5
f
;
return
rest_period
;
return
divisor
;
}
static
void
do_something
(
unsigned
x
)
void
ToneAlarm
::
start_note
(
unsigned
frequency
)
{
}
// check if circuit breaker is enabled
if
(
_cbrk
==
CBRK_UNINIT
)
{
_cbrk
=
circuit_breaker_enabled
(
"CBRK_BUZZER"
,
CBRK_BUZZER_KEY
);
}
if
(
_cbrk
!=
CBRK_OFF
)
{
return
;
}
void
ToneAlarm
::
start_note
(
unsigned
note
)
{
// compute the divisor
unsigned
divisor
=
note
_to_divisor
(
note
);
unsigned
divisor
=
frequency
_to_divisor
(
frequency
);
// pick the lowest prescaler value that we can use
// (note that the effective prescale value is 1 greater)
...
...
@@ -343,42 +216,30 @@ ToneAlarm::start_note(unsigned note)
unsigned
period
=
(
divisor
/
(
prescale
+
1
))
-
1
;
// Silence warning of unused var
do_something
(
period
);
UNUSED
(
period
);
PX4_DEBUG
(
"ToneAlarm::start_note %u"
,
period
);
}
void
ToneAlarm
::
stop_note
()
void
ToneAlarm
::
stop_note
()
{
}
void
ToneAlarm
::
start_tune
(
const
char
*
tune
)
void
ToneAlarm
::
next_note
()
{
PX4_DEBUG
(
"ToneAlarm::start_tune"
);
// kill any current playback
hrt_cancel
(
&
_note_call
);
// record the tune
_tune
=
tune
;
_next
=
tune
;
// initialise player state
_tempo
=
120
;
_note_length
=
4
;
_note_mode
=
MODE_NORMAL
;
_octave
=
4
;
_silence_length
=
0
;
_repeat
=
false
;
// otherwise command-line tunes repeat forever...
// schedule a callback to start playing
_note_call
=
{};
hrt_call_after
(
&
_note_call
,
0
,
(
hrt_callout
)
next_trampoline
,
this
);
}
if
(
!
_should_run
)
{
if
(
_tune_control_sub
>=
0
)
{
orb_unsubscribe
(
_tune_control_sub
);
}
_running
=
false
;
return
;
}
// subscribe to tune_control
if
(
_tune_control_sub
<
0
)
{
_tune_control_sub
=
orb_subscribe
(
ORB_ID
(
tune_control
));
}
void
ToneAlarm
::
next_note
()
{
// do we have an inter-note gap to wait for?
if
(
_silence_length
>
0
)
{
stop_note
();
...
...
@@ -388,347 +249,58 @@ ToneAlarm::next_note()
return
;
}
// make sure we still have a tune - may be removed by the write / ioctl handler
if
((
_next
==
nullptr
)
||
(
_tune
==
nullptr
))
{
stop_note
();
return
;
}
// parse characters out of the string until we have resolved a note
unsigned
note
=
0
;
unsigned
note_length
=
_note_length
;
unsigned
duration
;
while
(
note
==
0
)
{
// we always need at least one character from the string
int
c
=
next_char
();
if
(
c
==
0
)
{
goto
tune_end
;
}
_next
++
;
switch
(
c
)
{
case
'L'
:
// select note length
_note_length
=
next_number
();
if
(
_note_length
<
1
)
{
goto
tune_error
;
}
break
;
case
'O'
:
// select octave
_octave
=
next_number
();
if
(
_octave
>
6
)
{
_octave
=
6
;
}
break
;
case
'<'
:
// decrease octave
if
(
_octave
>
0
)
{
_octave
--
;
}
break
;
case
'>'
:
// increase octave
if
(
_octave
<
6
)
{
_octave
++
;
}
break
;
case
'M'
:
// select inter-note gap
c
=
next_char
();
if
(
c
==
0
)
{
goto
tune_error
;
}
_next
++
;
switch
(
c
)
{
case
'N'
:
_note_mode
=
MODE_NORMAL
;
break
;
case
'L'
:
_note_mode
=
MODE_LEGATO
;
break
;
case
'S'
:
_note_mode
=
MODE_STACCATO
;
break
;
case
'F'
:
_repeat
=
false
;
break
;
case
'B'
:
_repeat
=
true
;
break
;
default:
goto
tune_error
;
}
break
;
case
'P'
:
// pause for a note length
stop_note
();
_note_call
=
{};
hrt_call_after
(
&
_note_call
,
(
hrt_abstime
)
rest_duration
(
next_number
(),
next_dots
()),
(
hrt_callout
)
next_trampoline
,
this
);
return
;
case
'T'
:
{
// change tempo
unsigned
nt
=
next_number
();
if
((
nt
>=
32
)
&&
(
nt
<=
255
))
{
_tempo
=
nt
;
}
else
{
goto
tune_error
;
}
break
;
}
case
'N'
:
// play an arbitrary note
note
=
next_number
();
if
(
note
>
84
)
{
goto
tune_error
;
}
if
(
note
==
0
)
{
// this is a rest - pause for the current note length
_note_call
=
{};
hrt_call_after
(
&
_note_call
,
(
hrt_abstime
)
rest_duration
(
_note_length
,
next_dots
()),
(
hrt_callout
)
next_trampoline
,
this
);
return
;
}
break
;
case
'A'
...
'G'
:
// play a note in the current octave
note
=
_note_tab
[
c
-
'A'
]
+
(
_octave
*
12
)
+
1
;
c
=
next_char
();
// check for updates
bool
updated
=
false
;
orb_check
(
_tune_control_sub
,
&
updated
);
switch
(
c
)
{
case
'#'
:
// up a semitone
case
'+'
:
if
(
note
<
84
)
{
note
++
;
}
if
(
updated
)
{
orb_copy
(
ORB_ID
(
tune_control
),
_tune_control_sub
,
&
_tune
);
_play_tone
=
_tunes
.
set_control
(
_tune
)
==
0
;
}
_next
++
;
break
;
unsigned
frequency
=
0
;
unsigned
duration
=
0
;
case
'-'
:
// down a semitone
if
(
note
>
1
)
{
note
--
;
}
if
(
_play_tone
)
{
_play_tone
=
false
;
int
parse_ret_val
=
_tunes
.
get_next_tune
(
frequency
,
duration
,
_silence_length
);
_next
++
;
break
;
if
(
parse_ret_val
>=
0
)
{
// a frequency of 0 correspond to stop_note
if
(
frequency
>
0
)
{
// start playing the note
start_note
(
frequency
);
default
:
// 0 / no next char here is OK
break
;
}
else
{
stop_note
();
}
// shorthand length notation
note_length
=
next_number
();
if
(
note_length
==
0
)
{
note_length
=
_note_length
;
if
(
parse_ret_val
>
0
)
{
// continue playing
_play_tone
=
true
;
}
break
;
default
:
goto
tune_error
;
}
}
// compute the duration of the note and the following silence (if any)
duration
=
note_duration
(
_silence_length
,
note_length
,
next_dots
());
// start playing the note
start_note
(
note
);
// and arrange a callback when the note should stop
_note_call
=
{};
hrt_call_after
(
&
_note_call
,
(
hrt_abstime
)
duration
,
(
hrt_callout
)
next_trampoline
,
this
);
return
;
// tune looks bad (unexpected EOF, bad character, etc.)
tune_error
:
PX4_ERR
(
"tune error
\n
"
);
_repeat
=
false
;
// don't loop on error
// stop (and potentially restart) the tune
tune_end
:
stop_note
();
if
(
_repeat
)
{
start_tune
(
_tune
);
}
else
{
_tune
=
nullptr
;
_default_tune_number
=
0
;
}
}
int
ToneAlarm
::
next_char
()
{
while
(
isspace
(
*
_next
))
{
_next
++
;
}
return
toupper
(
*
_next
);
}
unsigned
ToneAlarm
::
next_number
()
{
unsigned
number
=
0
;
int
c
;
for
(;;)
{
c
=
next_char
();
if
(
!
isdigit
(
c
))
{
return
number
;
}
_next
++
;
number
=
(
number
*
10
)
+
(
c
-
'0'
);
}
}
unsigned
ToneAlarm
::
next_dots
()
{
unsigned
dots
=
0
;
while
(
next_char
()
==
'.'
)
{
_next
++
;
dots
++
;
// schedule a call with the tunes max interval
duration
=
_tunes
.
get_maximum_update_interval
();
// stop playing the last note after the duration elapsed
stop_note
();
}
return
dots
;
// and arrange a callback when the note should stop
assert
(
duration
!=
0
);
_note_call
=
{};
hrt_call_after
(
&
_note_call
,
(
hrt_abstime
)
duration
,
(
hrt_callout
)
next_trampoline
,
this
);
}
void
ToneAlarm
::
next_trampoline
(
void
*
arg
)
void
ToneAlarm
::
next_trampoline
(
void
*
arg
)
{
ToneAlarm
*
ta
=
(
ToneAlarm
*
)
arg
;
ta
->
next_note
();
}
int
ToneAlarm
::
devIOCTL
(
unsigned
long
cmd
,
unsigned
long
arg
)
{
int
result
=
OK
;
PX4_DEBUG
(
"ToneAlarm::devIOCTL %i %lu"
,
cmd
,
arg
);
/* decide whether to increase the alarm level to cmd or leave it alone */
switch
(
cmd
)
{
case
TONE_SET_ALARM
:
if
(
arg
<
TONE_NUMBER_OF_TUNES
)
{
if
(
arg
==
TONE_STOP_TUNE
)
{
// stop the tune
_tune
=
nullptr
;
_next
=
nullptr
;
_repeat
=
false
;
_default_tune_number
=
0
;
}
else
{
/* always interrupt alarms, unless they are repeating and already playing */
if
(
!
(
_repeat
&&
_default_tune_number
==
arg
))
{
/* play the selected tune */
_default_tune_number
=
arg
;
start_tune
(
_default_tunes
[
arg
]);
PX4_INFO
(
"%s"
,
_tune_names
[
arg
]);
}
}
}
else
{
result
=
-
EINVAL
;
}
break
;
default
:
result
=
-
ENOTTY
;
break
;
}
/* give it to the superclass if we didn't like it */
if
(
result
==
-
ENOTTY
)
{
result
=
VirtDevObj
::
devIOCTL
(
cmd
,
arg
);
}
return
result
;
}
ssize_t
ToneAlarm
::
devWrite
(
const
void
*
buffer
,
size_t
len
)
{
// sanity-check the buffer for length and nul-termination
if
(
len
>
_tune_max
)
{
return
-
EFBIG
;
}
// if we have an existing user tune, free it
if
(
_user_tune
!=
nullptr
)
{
// if we are playing the user tune, stop
if
(
_tune
==
_user_tune
)
{
_tune
=
nullptr
;
_next
=
nullptr
;
}
// free the old user tune
free
((
void
*
)
_user_tune
);
_user_tune
=
nullptr
;
}
const
char
*
buf
=
reinterpret_cast
<
const
char
*>
(
buffer
);
// if the new tune is empty, we're done
if
(
buf
[
0
]
==
'\0'
)
{
return
OK
;
}
// allocate a copy of the new tune
_user_tune
=
strndup
(
buf
,
len
);
if
(
_user_tune
==
nullptr
)
{
return
-
ENOMEM
;
}
// and play it
start_tune
(
_user_tune
);
return
len
;
}
/**
* Local functions in support of the shell command.
*/
...
...
@@ -737,147 +309,59 @@ namespace
ToneAlarm
*
g_dev
;
int
play_tune
(
unsigned
tune
)
{
int
ret
;
DevHandle
h
;
DevMgr
::
getHandle
(
TONEALARM0_DEVICE_PATH
,
h
);
if
(
!
h
.
isValid
())
{
PX4_WARN
(
"Error: failed to open %s (%d)"
,
TONEALARM0_DEVICE_PATH
,
h
.
getError
());
return
1
;
}
ret
=
h
.
ioctl
(
TONE_SET_ALARM
,
tune
);
if
(
ret
!=
0
)
{
PX4_WARN
(
"TONE_SET_ALARM"
);
return
1
;
}
}
// namespace
return
ret
;
}
void
tone_alarm_usage
();
int
play_string
(
const
char
*
str
,
bool
free_buffer
)
void
tone_alarm_usage
()
{
int
ret
;
DevHandle
h
;
DevMgr
::
getHandle
(
TONEALARM0_DEVICE_PATH
,
h
);
if
(
!
h
.
isValid
())
{
PX4_WARN
(
"Error: failed to get handle to %s"
,
TONEALARM0_DEVICE_PATH
);
return
1
;
}
ret
=
h
.
write
(
str
,
strlen
(
str
)
+
1
);
DevMgr
::
releaseHandle
(
h
);
if
(
free_buffer
)
{
free
((
void
*
)
str
);
}
if
(
ret
<
0
)
{
PX4_WARN
(
"play tune"
);
return
1
;
}
return
ret
;
PX4_INFO
(
"missing command, try 'start', status, 'stop'"
);
}
}
// namespace
int
tone_alarm_main
(
int
argc
,
char
*
argv
[])
int
tone_alarm_main
(
int
argc
,
char
*
argv
[])
{
unsigned
tune
;
int
ret
=
1
;
/* start the driver lazily */
if
(
g_dev
==
nullptr
)
{
g_dev
=
new
ToneAlarm
;
if
(
g_dev
==
nullptr
)
{
PX4_WARN
(
"couldn't allocate the ToneAlarm driver"
);
return
1
;
}
if
(
g_dev
->
init
()
!=
OK
)
{
delete
g_dev
;
PX4_WARN
(
"ToneAlarm init failed"
);
return
1
;
}
}
if
(
argc
>
1
)
{
const
char
*
argv1
=
argv
[
1
];
if
(
!
strcmp
(
argv1
,
"start"
))
{
ret
=
play_tune
(
TONE_STOP_TUNE
);
}
else
if
(
!
strcmp
(
argv1
,
"stop"
))
{
ret
=
play_tune
(
TONE_STOP_TUNE
);
}
else
if
((
tune
=
strtol
(
argv1
,
nullptr
,
10
))
!=
0
)
{
ret
=
play_tune
(
tune
);
}
/* If it is a file name then load and play it as a string */
else
if
(
*
argv1
==
'/'
)
{
FILE
*
fd
=
fopen
(
argv1
,
"r"
);
int
sz
;
char
*
buffer
;
if
(
fd
==
nullptr
)
{
PX4_WARN
(
"couldn't open '%s'"
,
argv1
);
return
1
;
if
(
g_dev
!=
nullptr
)
{
PX4_ERR
(
"already started"
);
exit
(
1
);
}
fseek
(
fd
,
0
,
SEEK_END
);
sz
=
ftell
(
fd
);
fseek
(
fd
,
0
,
SEEK_SET
);
buffer
=
(
char
*
)
malloc
(
sz
+
1
);
if
(
g_dev
==
nullptr
)
{
g_dev
=
new
ToneAlarm
();
if
(
buffer
==
nullptr
)
{
PX4_WARN
(
"not enough memory memory"
);
fclose
(
fd
);
return
1
;
}
if
(
g_dev
==
nullptr
)
{
PX4_ERR
(
"couldn't allocate the ToneAlarm driver"
);
exit
(
1
);
}
// FIXME - Make GCC happy
if
(
fread
(
buffer
,
sz
,
1
,
fd
))
{
}
if
(
OK
!=
g_dev
->
init
())
{
delete
g_dev
;
g_dev
=
nullptr
;
PX4_ERR
(
"ToneAlarm init failed"
);
exit
(
1
);
}
}
/* terminate the string */
buffer
[
sz
]
=
0
;
ret
=
play_string
(
buffer
,
true
);
fclose
(
fd
);
exit
(
0
);
}
/* if it looks like a PLAY string... */
else
if
(
argv1
&&
(
strlen
(
argv1
)
>
2
))
{
if
(
*
argv1
==
'M'
)
{
ret
=
play_string
(
argv1
,
false
);
}
}
else
{
/* It might be a tune name */
for
(
tune
=
1
;
tune
<
TONE_NUMBER_OF_TUNES
;
tune
++
)
{
if
(
!
strcmp
(
g_dev
->
name
(
tune
),
argv1
))
{
ret
=
play_tune
(
tune
);
return
ret
;
}
}
if
(
!
strcmp
(
argv1
,
"stop"
))
{
delete
g_dev
;
g_dev
=
nullptr
;
exit
(
0
);
}
PX4_WARN
(
"unrecognized command, try 'start', 'stop', an alarm number or name, or a file name starting with a '/'"
);
ret
=
1
;
if
(
!
strcmp
(
argv1
,
"status"
))
{
g_dev
->
status
();
exit
(
0
);
}
}
return
ret
;
tone_alarm_usage
();
exit
(
0
);
}
This diff is collapsed.
Click to expand it.
Preview
0%
Loading
Try again
or
attach a new file
.
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Save comment
Cancel
Please
register
or
sign in
to comment