-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathREADME
211 lines (157 loc) · 8.43 KB
/
README
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
0) Purpose
This project is a central system of the various permutations of the
Paparazzi project developed at Joby Energy and Makani Power. It
attempts to collect the functionality I developed to generate a
complete interface to the LCM multicast messaging system based on a
high-level config file. A lot of the design decisions are based on
the way Paparazzi does things, but this one is in Python so that in
spite of its messiness, most programmers can extend and improve it.
1) Build LCM
LCM depends on the glib library, and, for the python interface, the
python headers. On a debian-ish system, you can say
sudo apt-get install libglib2.0-dev python-dev
to install these dependencies. Now enter the `upstream' directory and
run `./configure && make && sudo make install' to install the LCM
toolkit.
There are more dependencies to be had if you want to install the Java
or MATLAB interfaces, but then you've got bigger problems :)
2) Build LCM interface and shared configuration
In the top level directory, run `make' to generate the LCM interface
in C and python. The configuration of this interface is based on XML
files located in `../conf/'.
You can see how to write the XML files based on the example files
provided. The three main config files are:
- types.xml, where you define structs that you want to be shared by
the project and transmissible via LCM messaging
- telemetry.xml, where you define specific instances where you want to
send structs over LCM
- settings.xml, where you define specific instances where you want
structs to be settable over LCM
You can also pass an AIRCRAFT= parameter to the `make' command to take
in a paparazzi-style XML file that defines many #defined quantities
for use in the build.
3) Use the generated interface
3.1) Make
Your external project should include `conftron/includes'. This
exports a number of object files in $(EXTOBJ), along with additions to
INCLUDES and LDFLAGS, that you should hand to your compiler and linker
in order to get ahold of the LCM interface. In order to run conftron
from an external project, for example to rebuild with different
AIRCRAFT= parameters without having to leave your original directory,
simply call `$(MAKE) -C $(CONFTRON_DIR) AIRCRAFT=$(AIRCRAFT)'.
More on this last part: If you pass AIRCRAFT=foo to your `make' call,
then conftron will try to build a file of #defines called `foo.h' and
export it as `AIRFRAME_CONSTANTS'. So, from your external project
that uses conftron, you can simply say `#include AIRFRAME_CONSTANTS'
and pass an AIRCRAFT parameter to the build process to get access to
the configuration XML for your airframe.
Currently build dependencies aren't used, because I don't understand
them, I had time constraints and you don't have to rebuild very often
anyway. If you can get build dependencies working so that calling
`make' in conftron is just a normal part of the external project's
build process, I promise beer.
3.2) Types
To use the generated types, simply #include <(classname)_types.h>,
where (classname) is e.g. `ap' or `sim' or any class you've defined in
types.xml.
3.3) Telemetry
To e.g. activate and periodically send and receive LCM telemetry from
your main loop, #include <lcm_telemetry.h>.
To send a message you've defined in telemetry.xml, just #include the
correct telemetry header after you have declared the struct you intend
to send. For example, say you have defined a type `foobar_t' in
types.xml (in class `myclass'), and you've declared a message of type
`foobar_t' named `quux'. Then in the c module where `quux' is
located:
static foobar_t quux;
#include <telemetry/myclass_foobar_t_quux.h>
This will not work correctly (you will get a compiler error) if you
don't #include the telemetry file AFTER you have declared `quux'. I'm
sorry that this is a bit of a hack, but it makes telemetry mostly
painless from your end, and it avoids having to extern lots of crap.
`quux' will now be sent down the LCM link at the rates configured in
telemetry.xml, provided you periodically call myclass_telemetry_send()
at the correct rate.
3.4) Settings
Settings work the same way as telemetry, so
static foobar_t quux;
#include <settings/myclass_foobar_t_quux.h>
is all that's required to allow settings messages that get sent up to
modify `quux'.
3.5) LCM interface
Simply #include <lcm_telemetry.h> for the toplevel lcm interface. To
initialize all classes, you can call lcm_init(const char *provider).
To send telemetry from all classes, you can call
`telemetry_send(void)'. To initialize settings for all classes, you
can call `settings_init(const char *provider)'. (This will also
initialize telemetry for those classes.) To check for new settings
messages from all classes, you can call `settings_check(void)'.
The system intends you to #define DT to be the approximate period of
your periodic loop. telemetry_send() should be called in that loop.
settings_check() can be called in that loop or more slowly; it is
non-blocking.
3.6) LCM class-by-class
If you have a class named `myclass', you can call
`myclass_lcm_init(const char *provider)' to initialize telemetry for
only that class. Likewise, `myclass_telemetry_send()' will send
telemetry for `myclass' at the appropriate rates (if called in a loop
with period DT).
Same goes for settings: `myclass_settings_init(const char *provider)'
will initialize just the channels for `myclass'. Likewise,
`myclass_settings_check(void)' will check for new settings only in
`myclass', whereas `settings_check(void)' will check for new settings
messages in all classes.
Why would you call them separately? At Joby/Makani, we have an
autopilot that is designed and built separately from the simulator or
from the actual low-level harness that runs it on the flight computer.
The autopilot code is responsible for initializing and running
periodic functions for all the telemetry and settings in class `ap';
the simulator handles its own telemetry and settings in class `sim'.
No matter where the autopilot is running, it does the same thing.
3.7) Channels
LCM telemetry is on channels named <classname>_<typename>_<varname>;
for example, the code above sends telemetry on channel
`myclass_foobar_t_quux'. The settings system subscribes to almost the
same channel, but `_set' is appended to the telemetry channel name.
When it has successfully updated a structure, the modified structure
is returned for confirmation on a channel like the telemetry channel
but with `_ack' appended. You may specify a custom channel for
settings or telemetry in xml; please think hard before you do this,
because it's easy to mungle things up.
4) Platform-specific issues
4.1) Windows
I like to guzzle shit-covered glass shards, too! Let's go out
sometime.
4.2) Mac OS X
For some reason there are sometimes problems with parallel make on OS
X. If you're encountering problems when you call `make' in this
directory, try removing `-j' (or replacing with `-j <number of
processor cores your computer has>') from the `all' compile target in
the Makefile.
5) Examples
Enter the `conftron_example/' directory to find the example program.
Copy or link the conf/ folder you find there to the folder that
contains conftron/ (so relatively, it's ../../). Now type `make
conftron AIRCRAFT=testac && make AIRCRAFT=testac'. This builds a
small C program (test, from test.c) with a module (testmodule.[ch])
that sends periodic LCM telemetry and receives LCM settings, along
with python programs that can be used to monitor the sent telemetry
(test_telem.py) and send a settings message and confirm that the
messaging is working (test_settings.py). The test program has access
to #defined values from `conf/airframes/testac.xml' during the build
process. The makefile illustrates how to integrate conftron into the
build process of an external project.
6) Bugs
Present and largely unaccounted for. The code is a mess, too. Bug
reports, suggestions and patches are welcome.
7) Internals, for the too curious for their own good
- Paths to conf files are largely hardcoded, either at the top of
lcmgen.py or in the makefile.
- If you see weird linker errors, it might be because conftron passes
a library of stub telemetry and settings functions that don't do
anything and relies on the linker to grab the stubs for all the ones
you don't use in your code.
8) Thanks
Eric Parsonage came up with a good deal of the make and linker magic.
Greg Horn mercilessly bug-tested everything, helped refine the design,
pointed out what features were important and tested on OS X.