Compare commits
120 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
99c891e1cd | ||
|
|
378db51de9 | ||
|
|
8a83d70bb6 | ||
|
|
ba5ac30136 | ||
|
|
baf724ed6e | ||
|
|
af58d56138 | ||
|
|
13341602f0 | ||
|
|
84368738b4 | ||
|
|
0397b3a736 | ||
|
|
c41c96a96d | ||
|
|
a94687bd56 | ||
|
|
abcc9b96de | ||
|
|
f46460e53d | ||
|
|
23f3295439 | ||
|
|
b0b6eb9a97 | ||
|
|
84a329cca7 | ||
|
|
5613028678 | ||
|
|
a0cca80980 | ||
|
|
bed5d79db8 | ||
|
|
da51ebab38 | ||
|
|
0b977aee28 | ||
|
|
6ef8820770 | ||
|
|
e993dde355 | ||
|
|
627233f862 | ||
|
|
ce87234a51 | ||
|
|
e40fbd0ce2 | ||
|
|
0938609dc7 | ||
|
|
1a22350775 | ||
|
|
72b2cf49d5 | ||
|
|
63d602dd7a | ||
|
|
1119c77cca | ||
|
|
fbe33eac1b | ||
|
|
7cfcf5b63b | ||
|
|
94d24cbd28 | ||
|
|
be3d2be9d3 | ||
|
|
ad6bc02643 | ||
|
|
b91f4d3b6d | ||
|
|
28da7baf61 | ||
|
|
7516279132 | ||
|
|
a383c83a29 | ||
|
|
6392c4a97a | ||
|
|
cfb2e60310 | ||
|
|
41a9a95747 | ||
|
|
24e8569905 | ||
|
|
fb80b899e0 | ||
|
|
d095b81d7e | ||
|
|
28a6bf2230 | ||
|
|
fff7262d1b | ||
|
|
646fa46f6b | ||
|
|
f782f647cb | ||
|
|
32f29dc1a4 | ||
|
|
2cf1c7abb3 | ||
|
|
d752cce0cc | ||
|
|
aeec8e34eb | ||
|
|
34a81536c2 | ||
|
|
1c9b10a674 | ||
|
|
ab2f99ab59 | ||
|
|
5b6ef9c50e | ||
|
|
5ec6b5e665 | ||
|
|
85182ac2b8 | ||
|
|
455729fdb4 | ||
|
|
4eec63adfa | ||
|
|
e0db3bee38 | ||
|
|
bf803cf345 | ||
|
|
33319db1fa | ||
|
|
ba6e63b50b | ||
|
|
410fccf015 | ||
|
|
31d382dd86 | ||
|
|
0661aecccf | ||
|
|
0f83e8ed80 | ||
|
|
f718af7f0e | ||
|
|
4850b95029 | ||
|
|
2694f68b87 | ||
|
|
033e74a375 | ||
|
|
a24f039f1d | ||
|
|
6b52ad562b | ||
|
|
69cfc9e5fa | ||
|
|
1b54b3fa25 | ||
|
|
f794da916d | ||
|
|
ed6d09061b | ||
|
|
26a028ff66 | ||
|
|
2d365dcffe | ||
|
|
c22961e5ff | ||
|
|
9ad718cb85 | ||
|
|
172f6b173a | ||
|
|
8e629e3eea | ||
|
|
482bb8ed71 | ||
|
|
4ec6ff3f37 | ||
|
|
9ed41e50a1 | ||
|
|
344835cba8 | ||
|
|
654badd097 | ||
|
|
a8cd72e654 | ||
|
|
f4aaf0f4f3 | ||
|
|
1ed05a94dd | ||
|
|
e1e747969b | ||
|
|
48ea797a47 | ||
|
|
476f24f774 | ||
|
|
7a62229125 | ||
|
|
e7864b1e55 | ||
|
|
f72745a2e7 | ||
|
|
317ecc95cc | ||
|
|
d3700d5784 | ||
|
|
d84ed99996 | ||
|
|
82f3ab563a | ||
|
|
2fbebe102e | ||
|
|
fe7c06666f | ||
|
|
f520b57abe | ||
|
|
78f3f6e3b3 | ||
|
|
46ba00fca7 | ||
|
|
d2296fea76 | ||
|
|
645b148564 | ||
|
|
3207fdb43c | ||
|
|
c58a16e4df | ||
|
|
adeea474c6 | ||
|
|
fc006d43e2 | ||
|
|
776967038c | ||
|
|
93bfc5d258 | ||
|
|
d73cfe0c59 | ||
|
|
343935f98c | ||
|
|
886e592a20 |
@@ -4,8 +4,12 @@ root = true
|
||||
end_of_line = lf
|
||||
insert_final_newline = true
|
||||
|
||||
[*.{ino,cpp,c,h,hpp,sdf,world}]
|
||||
[*.{ino,cpp,c,h,hpp,sdf,world,json}]
|
||||
charset = utf-8
|
||||
indent_style = tab
|
||||
tab_width = 4
|
||||
trim_trailing_whitespace = true
|
||||
|
||||
[{*.yml,*.yaml,CMakeLists.txt}]
|
||||
indent_style = space
|
||||
indent_size = 2
|
||||
|
||||
2
.gitattributes
vendored
Normal file
@@ -0,0 +1,2 @@
|
||||
# https://github.com/github-linguist/linguist/blob/master/docs/overrides.md
|
||||
*.h linguist-language=C++
|
||||
49
.github/workflows/build.yml
vendored
@@ -7,39 +7,47 @@ on:
|
||||
branches: [ master ]
|
||||
|
||||
jobs:
|
||||
build:
|
||||
build_linux:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
- uses: actions/checkout@v4
|
||||
- name: Install Arduino CLI
|
||||
uses: arduino/setup-arduino-cli@v1.1.1
|
||||
run: curl -fsSL https://raw.githubusercontent.com/arduino/arduino-cli/master/install.sh | BINDIR=/usr/local/bin sh
|
||||
- name: Build firmware
|
||||
run: make
|
||||
- name: Check c_cpp_properties.json
|
||||
run: tools/check_c_cpp_properties.py
|
||||
|
||||
build_macos:
|
||||
runs-on: macos-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
- uses: actions/checkout@v4
|
||||
- name: Install Arduino CLI
|
||||
run: brew install arduino-cli
|
||||
- name: Build firmware
|
||||
run: make
|
||||
- name: Check c_cpp_properties.json
|
||||
run: tools/check_c_cpp_properties.py
|
||||
|
||||
build_windows:
|
||||
runs-on: windows-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
- uses: actions/checkout@v4
|
||||
- name: Install Arduino CLI
|
||||
run: choco install arduino-cli
|
||||
- name: Install Make
|
||||
run: choco install make
|
||||
- name: Build firmware
|
||||
run: make
|
||||
- name: Check c_cpp_properties.json
|
||||
run: python3 tools/check_c_cpp_properties.py
|
||||
|
||||
build_simulator:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v3
|
||||
- name: Install Arduino CLI
|
||||
uses: arduino/setup-arduino-cli@v1.1.1
|
||||
- uses: actions/checkout@v4
|
||||
- name: Install Gazebo
|
||||
run: curl -sSL http://get.gazebosim.org | sh
|
||||
- name: Install SDL2
|
||||
@@ -52,13 +60,22 @@ jobs:
|
||||
path: gazebo/build/*.so
|
||||
retention-days: 1
|
||||
|
||||
# build_simulator_macos:
|
||||
# runs-on: macos-latest
|
||||
# steps:
|
||||
# - uses: actions/checkout@v3
|
||||
# - name: Install Gazebo
|
||||
# run: brew tap osrf/simulation && brew install gazebo11
|
||||
# - name: Install SDL2
|
||||
# run: brew install sdl2
|
||||
# - name: Build simulator
|
||||
# run: make build_simulator
|
||||
build_simulator_macos:
|
||||
runs-on: macos-latest
|
||||
steps:
|
||||
- name: Install Arduino CLI
|
||||
run: brew install arduino-cli
|
||||
- uses: actions/checkout@v4
|
||||
- name: Clean up python binaries # Workaround for https://github.com/actions/setup-python/issues/577
|
||||
run: |
|
||||
rm -f /usr/local/bin/2to3*
|
||||
rm -f /usr/local/bin/idle3*
|
||||
rm -f /usr/local/bin/pydoc3*
|
||||
rm -f /usr/local/bin/python3*
|
||||
rm -f /usr/local/bin/python3*-config
|
||||
- name: Install Gazebo
|
||||
run: brew update && brew tap osrf/simulation && brew install gazebo11
|
||||
- name: Install SDL2
|
||||
run: brew install sdl2
|
||||
- name: Build simulator
|
||||
run: make build_simulator
|
||||
|
||||
33
.github/workflows/tools.yml
vendored
Normal file
@@ -0,0 +1,33 @@
|
||||
name: Build tools
|
||||
|
||||
on:
|
||||
push:
|
||||
branches: [ '*' ]
|
||||
pull_request:
|
||||
branches: [ master ]
|
||||
|
||||
jobs:
|
||||
csv_to_ulog:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- name: Build csv_to_ulog
|
||||
run: cd tools/csv_to_ulog && mkdir build && cd build && cmake .. && make
|
||||
- name: Test csv_to_ulog
|
||||
run: |
|
||||
cd tools/csv_to_ulog/build
|
||||
echo -e "t,x,y,z\n0,1,2,3\n1,4,5,6" > log.csv
|
||||
./csv_to_ulog log.csv
|
||||
test $(stat -c %s log.ulg) -eq 196
|
||||
python_tools:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/checkout@v4
|
||||
- name: Install Python dependencies
|
||||
run: pip install -r tools/requirements.txt
|
||||
- name: Test csv_to_mcap tool
|
||||
run: |
|
||||
cd tools
|
||||
echo -e "t,x,y,z\n0,1,2,3\n1,4,5,6" > log.csv
|
||||
./csv_to_mcap.py log.csv
|
||||
test $(stat -c %s log.mcap) -eq 883
|
||||
9
.gitignore
vendored
@@ -1,5 +1,12 @@
|
||||
*.hex
|
||||
*.elf
|
||||
gazebo/build/
|
||||
build/
|
||||
tools/log/
|
||||
.dependencies
|
||||
.vscode/*
|
||||
!.vscode/settings.json
|
||||
!.vscode/c_cpp_properties.json
|
||||
!.vscode/tasks.json
|
||||
!.vscode/launch.json
|
||||
!.vscode/extensions.json
|
||||
!.vscode/intellisense.h
|
||||
|
||||
147
.vscode/c_cpp_properties.json
vendored
Normal file
@@ -0,0 +1,147 @@
|
||||
{
|
||||
"configurations": [
|
||||
{
|
||||
"name": "Linux",
|
||||
"includePath": [
|
||||
"${workspaceFolder}/flix",
|
||||
"${workspaceFolder}/gazebo",
|
||||
"~/.arduino15/packages/esp32/hardware/esp32/3.0.3/cores/esp32",
|
||||
"~/.arduino15/packages/esp32/hardware/esp32/3.0.3/libraries/**",
|
||||
"~/.arduino15/packages/esp32/hardware/esp32/3.0.3/variants/d1_mini32",
|
||||
"~/.arduino15/packages/esp32/tools/esp32-arduino-libs/idf-release_v5.1-dc859c1e67/esp32/**",
|
||||
"~/.arduino15/packages/esp32/tools/esp32-arduino-libs/idf-release_v5.1-dc859c1e67/esp32/dio_qspi/include",
|
||||
"~/Arduino/libraries/**",
|
||||
"/usr/include/**"
|
||||
],
|
||||
"forcedInclude": [
|
||||
"${workspaceFolder}/.vscode/intellisense.h",
|
||||
"~/.arduino15/packages/esp32/hardware/esp32/3.0.3/cores/esp32/Arduino.h",
|
||||
"~/.arduino15/packages/esp32/hardware/esp32/3.0.3/variants/d1_mini32/pins_arduino.h",
|
||||
"${workspaceFolder}/flix/cli.ino",
|
||||
"${workspaceFolder}/flix/control.ino",
|
||||
"${workspaceFolder}/flix/estimate.ino",
|
||||
"${workspaceFolder}/flix/flix.ino",
|
||||
"${workspaceFolder}/flix/imu.ino",
|
||||
"${workspaceFolder}/flix/led.ino",
|
||||
"${workspaceFolder}/flix/log.ino",
|
||||
"${workspaceFolder}/flix/mavlink.ino",
|
||||
"${workspaceFolder}/flix/motors.ino",
|
||||
"${workspaceFolder}/flix/rc.ino",
|
||||
"${workspaceFolder}/flix/time.ino",
|
||||
"${workspaceFolder}/flix/util.ino",
|
||||
"${workspaceFolder}/flix/wifi.ino"
|
||||
],
|
||||
"compilerPath": "~/.arduino15/packages/esp32/tools/esp-x32/2302/bin/xtensa-esp32-elf-g++",
|
||||
"cStandard": "c11",
|
||||
"cppStandard": "c++17",
|
||||
"defines": [
|
||||
"F_CPU=240000000L",
|
||||
"ARDUINO=10607",
|
||||
"ARDUINO_D1_MINI32",
|
||||
"ARDUINO_ARCH_ESP32",
|
||||
"ARDUINO_BOARD=D1_MINI32",
|
||||
"ARDUINO_VARIANT=d1_mini32",
|
||||
"ARDUINO_PARTITION_default",
|
||||
"ESP32",
|
||||
"CORE_DEBUG_LEVEL=0",
|
||||
"ARDUINO_USB_CDC_ON_BOOT="
|
||||
]
|
||||
},
|
||||
{
|
||||
"name": "Mac",
|
||||
"includePath": [
|
||||
"${workspaceFolder}/flix",
|
||||
"${workspaceFolder}/gazebo",
|
||||
"~/Library/Arduino15/packages/esp32/hardware/esp32/3.0.3/cores/esp32",
|
||||
"~/Library/Arduino15/packages/esp32/hardware/esp32/3.0.3/libraries/**",
|
||||
"~/Library/Arduino15/packages/esp32/hardware/esp32/3.0.3/variants/d1_mini32",
|
||||
"~/Library/Arduino15/packages/esp32/tools/esp32-arduino-libs/idf-release_v5.1-dc859c1e67/esp32/include/**",
|
||||
"~/Library/Arduino15/packages/esp32/tools/esp32-arduino-libs/idf-release_v5.1-dc859c1e67/esp32/dio_qspi/include",
|
||||
"~/Documents/Arduino/libraries/**",
|
||||
"/opt/homebrew/include/**"
|
||||
],
|
||||
"forcedInclude": [
|
||||
"${workspaceFolder}/.vscode/intellisense.h",
|
||||
"~/Library/Arduino15/packages/esp32/hardware/esp32/3.0.3/cores/esp32/Arduino.h",
|
||||
"~/Library/Arduino15/packages/esp32/hardware/esp32/3.0.3/variants/d1_mini32/pins_arduino.h",
|
||||
"${workspaceFolder}/flix/flix.ino",
|
||||
"${workspaceFolder}/flix/cli.ino",
|
||||
"${workspaceFolder}/flix/control.ino",
|
||||
"${workspaceFolder}/flix/estimate.ino",
|
||||
"${workspaceFolder}/flix/imu.ino",
|
||||
"${workspaceFolder}/flix/led.ino",
|
||||
"${workspaceFolder}/flix/log.ino",
|
||||
"${workspaceFolder}/flix/mavlink.ino",
|
||||
"${workspaceFolder}/flix/motors.ino",
|
||||
"${workspaceFolder}/flix/rc.ino",
|
||||
"${workspaceFolder}/flix/time.ino",
|
||||
"${workspaceFolder}/flix/util.ino",
|
||||
"${workspaceFolder}/flix/wifi.ino"
|
||||
],
|
||||
"compilerPath": "~/Library/Arduino15/packages/esp32/tools/esp-x32/2302/bin/xtensa-esp32-elf-g++",
|
||||
"cStandard": "c11",
|
||||
"cppStandard": "c++17",
|
||||
"defines": [
|
||||
"F_CPU=240000000L",
|
||||
"ARDUINO=10607",
|
||||
"ARDUINO_D1_MINI32",
|
||||
"ARDUINO_ARCH_ESP32",
|
||||
"ARDUINO_BOARD=D1_MINI32",
|
||||
"ARDUINO_VARIANT=d1_mini32",
|
||||
"ARDUINO_PARTITION_default",
|
||||
"ARDUINO_FQBN=esp32:esp32:d1_mini32",
|
||||
"ESP32",
|
||||
"CORE_DEBUG_LEVEL=0",
|
||||
"ARDUINO_USB_CDC_ON_BOOT="
|
||||
]
|
||||
},
|
||||
{
|
||||
"name": "Win32",
|
||||
"includePath": [
|
||||
"${workspaceFolder}/flix",
|
||||
"${workspaceFolder}/gazebo",
|
||||
"~/AppData/Local/Arduino15/packages/esp32/hardware/esp32/3.0.3/cores/esp32",
|
||||
"~/AppData/Local/Arduino15/packages/esp32/hardware/esp32/3.0.3/libraries/**",
|
||||
"~/AppData/Local/Arduino15/packages/esp32/hardware/esp32/3.0.3/variants/d1_mini32",
|
||||
"~/AppData/Local/Arduino15/packages/esp32/tools/esp32-arduino-libs/idf-release_v5.1-dc859c1e67/esp32/**",
|
||||
"~/AppData/Local/Arduino15/packages/esp32/tools/esp32-arduino-libs/idf-release_v5.1-dc859c1e67/esp32/dio_qspi/include",
|
||||
"~/Documents/Arduino/libraries/**"
|
||||
],
|
||||
"forcedInclude": [
|
||||
"${workspaceFolder}/.vscode/intellisense.h",
|
||||
"~/AppData/Local/Arduino15/packages/esp32/hardware/esp32/3.0.3/cores/esp32/Arduino.h",
|
||||
"~/AppData/Local/Arduino15/packages/esp32/hardware/esp32/3.0.3/variants/d1_mini32/pins_arduino.h",
|
||||
"${workspaceFolder}/flix/cli.ino",
|
||||
"${workspaceFolder}/flix/control.ino",
|
||||
"${workspaceFolder}/flix/estimate.ino",
|
||||
"${workspaceFolder}/flix/flix.ino",
|
||||
"${workspaceFolder}/flix/imu.ino",
|
||||
"${workspaceFolder}/flix/led.ino",
|
||||
"${workspaceFolder}/flix/log.ino",
|
||||
"${workspaceFolder}/flix/mavlink.ino",
|
||||
"${workspaceFolder}/flix/motors.ino",
|
||||
"${workspaceFolder}/flix/rc.ino",
|
||||
"${workspaceFolder}/flix/time.ino",
|
||||
"${workspaceFolder}/flix/util.ino",
|
||||
"${workspaceFolder}/flix/wifi.ino"
|
||||
],
|
||||
"compilerPath": "~/AppData/Local/Arduino15/packages/esp32/tools/esp-x32/2302/bin/xtensa-esp32-elf-g++.exe",
|
||||
"cStandard": "c11",
|
||||
"cppStandard": "c++17",
|
||||
"defines": [
|
||||
"F_CPU=240000000L",
|
||||
"ARDUINO=10607",
|
||||
"ARDUINO_D1_MINI32",
|
||||
"ARDUINO_ARCH_ESP32",
|
||||
"ARDUINO_BOARD=D1_MINI32",
|
||||
"ARDUINO_VARIANT=d1_mini32",
|
||||
"ARDUINO_PARTITION_default",
|
||||
"ARDUINO_FQBN=esp32:esp32:d1_mini32",
|
||||
"ESP32",
|
||||
"CORE_DEBUG_LEVEL=0",
|
||||
"ARDUINO_USB_CDC_ON_BOOT="
|
||||
]
|
||||
}
|
||||
],
|
||||
"version": 4
|
||||
}
|
||||
10
.vscode/extensions.json
vendored
Normal file
@@ -0,0 +1,10 @@
|
||||
{
|
||||
// See https://go.microsoft.com/fwlink/?LinkId=827846 to learn about workspace recommendations.
|
||||
"recommendations": [
|
||||
"ms-vscode.cpptools",
|
||||
"twxs.cmake",
|
||||
"ms-vscode.cmake-tools",
|
||||
"ms-python.python"
|
||||
],
|
||||
"unwantedRecommendations": []
|
||||
}
|
||||
5
.vscode/intellisense.h
vendored
Normal file
@@ -0,0 +1,5 @@
|
||||
#ifdef __INTELLISENSE__
|
||||
#pragma diag_suppress 144, 513
|
||||
// diag 144: a value of type "enum <unnamed>" cannot be used to initialize an entity of type "enum <unnamed>"C/C++
|
||||
// diag 513: a value of type "enum <unnamed>" cannot be assigned to an entity of type "enum <unnamed>"C/C++
|
||||
#endif
|
||||
25
.vscode/launch.json
vendored
Normal file
@@ -0,0 +1,25 @@
|
||||
{
|
||||
"version": "0.2.0",
|
||||
"configurations": [
|
||||
{
|
||||
"name": "Debug simulation",
|
||||
"type": "cppdbg",
|
||||
"request": "launch",
|
||||
"program": "/usr/bin/gzserver",
|
||||
"osx": {
|
||||
"program": "/opt/homebrew/bin/gzserver",
|
||||
"MIMode": "lldb",
|
||||
},
|
||||
"args": ["--verbose", "${workspaceFolder}/gazebo/flix.world"],
|
||||
"stopAtEntry": false,
|
||||
"cwd": "${fileDirname}",
|
||||
"environment": [
|
||||
{"name": "GAZEBO_MODEL_PATH", "value": "${workspaceFolder}/gazebo/models"},
|
||||
{"name": "GAZEBO_PLUGIN_PATH", "value": "${workspaceFolder}/gazebo/build"}
|
||||
],
|
||||
"MIMode": "gdb",
|
||||
"preLaunchTask": "Build simulator",
|
||||
"externalConsole": true,
|
||||
},
|
||||
]
|
||||
}
|
||||
13
.vscode/settings.json
vendored
Normal file
@@ -0,0 +1,13 @@
|
||||
{
|
||||
"C_Cpp.intelliSenseEngineFallback": "enabled",
|
||||
"files.associations": {
|
||||
"*.sdf": "xml",
|
||||
"*.ino": "cpp",
|
||||
"*.h": "cpp"
|
||||
},
|
||||
"C_Cpp.vcFormat.newLine.beforeOpenBrace.function": "newLine",
|
||||
"C_Cpp.vcFormat.newLine.beforeOpenBrace.block": "sameLine",
|
||||
"C_Cpp.vcFormat.newLine.beforeOpenBrace.lambda": "sameLine",
|
||||
"C_Cpp.vcFormat.newLine.beforeOpenBrace.namespace": "sameLine",
|
||||
"C_Cpp.vcFormat.newLine.beforeOpenBrace.type": "sameLine"
|
||||
}
|
||||
31
.vscode/tasks.json
vendored
Normal file
@@ -0,0 +1,31 @@
|
||||
{
|
||||
"tasks": [
|
||||
{
|
||||
"label": "Build firmware",
|
||||
"type": "shell",
|
||||
"command": "make",
|
||||
"problemMatcher": [ "$gcc" ],
|
||||
"presentation": { "clear": true, "showReuseMessage": false },
|
||||
},
|
||||
{
|
||||
"label": "Upload firmware",
|
||||
"type": "shell",
|
||||
"command": "make upload",
|
||||
"problemMatcher": [ "$gcc" ],
|
||||
"presentation": { "clear": true, "showReuseMessage": false }
|
||||
},
|
||||
{
|
||||
"label": "Build simulator",
|
||||
"type": "shell",
|
||||
"command": "make build_simulator",
|
||||
"problemMatcher": [ "$gcc" ],
|
||||
"presentation": { "clear": true, "showReuseMessage": false }
|
||||
},
|
||||
{
|
||||
"label": "Clean",
|
||||
"type": "shell",
|
||||
"command": "make clean",
|
||||
}
|
||||
],
|
||||
"version": "2.0.0"
|
||||
}
|
||||
14
Makefile
@@ -1,5 +1,5 @@
|
||||
BOARD = esp32:esp32:d1_mini32
|
||||
PORT := $(wildcard /dev/serial/by-id/usb-Silicon_Labs_CP2104_USB_to_UART_Bridge_Controller_* /dev/serial/by-id/usb-1a86_USB_Single_Serial_* /dev/cu.usbserial-*)
|
||||
PORT := $(wildcard /dev/serial/by-id/usb-Silicon_Labs_CP21* /dev/serial/by-id/usb-1a86_USB_Single_Serial_* /dev/cu.usbserial-*)
|
||||
PORT := $(strip $(PORT))
|
||||
|
||||
build: .dependencies
|
||||
@@ -13,16 +13,17 @@ monitor:
|
||||
|
||||
dependencies .dependencies:
|
||||
arduino-cli core update-index --config-file arduino-cli.yaml
|
||||
arduino-cli core install esp32:esp32@2.0.11 --config-file arduino-cli.yaml
|
||||
arduino-cli lib install "Bolder Flight Systems SBUS"@1.0.1
|
||||
arduino-cli lib install --git-url https://github.com/okalachev/MPU9250.git --config-file arduino-cli.yaml
|
||||
arduino-cli core install esp32:esp32@3.0.3 --config-file arduino-cli.yaml
|
||||
arduino-cli lib update-index
|
||||
arduino-cli lib install "FlixPeriph"
|
||||
arduino-cli lib install "MAVLink"@2.0.10
|
||||
touch .dependencies
|
||||
|
||||
gazebo/build cmake: gazebo/CMakeLists.txt
|
||||
mkdir -p gazebo/build
|
||||
cd gazebo/build && cmake ..
|
||||
|
||||
build_simulator: gazebo/build
|
||||
build_simulator: .dependencies gazebo/build
|
||||
make -C gazebo/build
|
||||
|
||||
simulator: build_simulator
|
||||
@@ -36,9 +37,6 @@ log:
|
||||
plot:
|
||||
plotjuggler -d $(shell ls -t tools/log/*.csv | head -n1)
|
||||
|
||||
docs:
|
||||
for FILE in docs/*.d2; do d2 $$FILE; done
|
||||
|
||||
clean:
|
||||
rm -rf gazebo/build flix/build flix/cache .dependencies
|
||||
|
||||
|
||||
150
README.md
@@ -1,58 +1,144 @@
|
||||
# flix
|
||||
# Flix
|
||||
|
||||
**flix** (*flight + X*) — making an open source ESP32-based quadcopter from scratch.
|
||||
**Flix** (*flight + X*) — making an open source ESP32-based quadcopter from scratch.
|
||||
|
||||
<img src="docs/img/flix.jpg" width=500>
|
||||
<table>
|
||||
<tr>
|
||||
<td align=center><strong>Version 1</strong> (3D-printed frame)</td>
|
||||
<td align=center><strong>Version 0</strong></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td><img src="docs/img/flix1.jpg" width=500 alt="Flix quadcopter"></td>
|
||||
<td><img src="docs/img/flix.jpg" width=500 alt="Flix quadcopter"></td>
|
||||
</tr>
|
||||
</table>
|
||||
|
||||
## Features
|
||||
|
||||
* Simple and clear Arduino based source code.
|
||||
* Simple and clean Arduino based source code.
|
||||
* Acro and Stabilized flight using remote control.
|
||||
* Precise simulation using Gazebo.
|
||||
* In-RAM logging.
|
||||
* [In-RAM logging](docs/log.md).
|
||||
* Command line interface through USB port.
|
||||
* Wi-Fi support.
|
||||
* ESCs with reverse mode support.
|
||||
* *Textbook and videos for students on writing a flight controller\*.*
|
||||
* *MAVLink support\*.*
|
||||
* *Completely 3D-printed frame*.*
|
||||
* *Position control and autonomous flights using external camera\**.
|
||||
* MAVLink support.
|
||||
* Control using mobile phone (with QGroundControl app).
|
||||
* Completely 3D-printed frame.
|
||||
* *Textbook and videos for students on writing a flight controller¹.*
|
||||
* *Position control and autonomous flights using external camera¹*.
|
||||
* [Building and running instructions](docs/build.md).
|
||||
|
||||
*\* — planned.*
|
||||
*¹ — planned.*
|
||||
|
||||
## It actually flies
|
||||
|
||||
See detailed demo video (for version 0): https://youtu.be/8GzzIQ3C6DQ.
|
||||
|
||||
<a href="https://youtu.be/8GzzIQ3C6DQ"><img width=500 src="https://i3.ytimg.com/vi/8GzzIQ3C6DQ/maxresdefault.jpg"></a>
|
||||
|
||||
See YouTube demo video: https://youtu.be/8GzzIQ3C6DQ.
|
||||
Version 1 test flight: https://t.me/opensourcequadcopter/42.
|
||||
|
||||
<a href="https://t.me/opensourcequadcopter/42"><img width=500 src="docs/img/flight-video.jpg"></a>
|
||||
|
||||
## Simulation
|
||||
|
||||
Simulation in Gazebo using a plugin that runs original Arduino code is implemented:
|
||||
The simulator is implemented using Gazebo and runs the original Arduino code:
|
||||
|
||||
<img src="docs/img/simulator.png" width=500>
|
||||
<img src="docs/img/simulator.png" width=500 alt="Flix simulator">
|
||||
|
||||
## Schematics
|
||||
See [instructions on running the simulation](docs/build.md).
|
||||
|
||||
<img src="docs/img/schematics.svg" width=800>
|
||||
## Components (version 1)
|
||||
|
||||
|Type|Part|Image|Quantity|
|
||||
|-|-|:-:|:-:|
|
||||
|Microcontroller board|ESP32 Mini|<img src="docs/img/esp32.jpg" width=100>|1|
|
||||
|IMU and barometer² board|GY-91 (or other MPU-9250 board)|<img src="docs/img/gy-91.jpg" width=100>|1|
|
||||
|Motor|8520 3.7V brushed motor (**shaft 0.8mm!**)|<img src="docs/img/motor.jpeg" width=100>|4|
|
||||
|Propeller|Hubsan 55 mm|<img src="docs/img/prop.jpg" width=100>|4|
|
||||
|MOSFET (transistor)|100N03A or [compatible](https://t.me/opensourcequadcopter/33)|<img src="docs/img/100n03a.jpg" width=100>|4|
|
||||
|Pull-down resistor|10 kΩ|<img src="docs/img/resistor10k.jpg" width=100>|4|
|
||||
|3.7V Li-Po battery|LW 952540 (or any compatible by the size)|<img src="docs/img/battery.jpg" width=100>|1|
|
||||
|Li-Po Battery charger|Any|<img src="docs/img/charger.jpg" width=100>|1|
|
||||
|Screws for IMU board mounting|M3x5|<img src="docs/img/screw-m3.jpg" width=100>|2|
|
||||
|Screws for frame assembly|M1.4x5|<img src="docs/img/screw-m1.4.jpg" height=30 align=center>|4|
|
||||
|Frame bottom part|3D printed: [`flix-frame.stl`](docs/assets/flix-frame.stl)|<img src="docs/img/frame1.jpg" width=100>|1|
|
||||
|Frame top part|3D printed: [`esp32-holder.stl`](docs/assets/esp32-holder.stl)|<img src="docs/img/esp32-holder.jpg" width=100>|1|
|
||||
|Washer for IMU board mounting|3D printed: [`washer-m3.stl`](docs/assets/washer-m3.stl)|<img src="docs/img/washer-m3.jpg" width=100>|1|
|
||||
|*RC transmitter (optional)*|*KINGKONG TINY X8 or other³*|<img src="docs/img/tx.jpg" width=100>|1|
|
||||
|*RC receiver (optional)*|*DF500 or other³*|<img src="docs/img/rx.jpg" width=100>|1|
|
||||
|Wires|28 AWG recommended|<img src="docs/img/wire-28awg.jpg" width=100>||
|
||||
|Tape, double-sided tape||||
|
||||
|
||||
*² — barometer is not used for now.*<br>
|
||||
*³ — you may use any transmitter-receiver pair with SBUS interface.*
|
||||
|
||||
Tools required for assembly:
|
||||
|
||||
* 3D printer.
|
||||
* Soldering iron.
|
||||
* Solder wire (with flux).
|
||||
* Screwdrivers.
|
||||
* Multimeter.
|
||||
|
||||
Feel free to modify the design and or code, and create your own improved versions of Flix! Send your results to the [official Telegram chat](https://t.me/opensourcequadcopterchat), or directly to the author ([E-mail](mailto:okalachev@gmail.com), [Telegram](https://t.me/okalachev)).
|
||||
|
||||
## Schematics (version 1)
|
||||
|
||||
### Simplified connection diagram
|
||||
|
||||
<img src="docs/img/schematics1.svg" width=800 alt="Flix version 1 schematics">
|
||||
|
||||
Motor connection scheme:
|
||||
|
||||
<img src="docs/img/mosfet-connection.png" height=400 alt="MOSFET connection scheme">
|
||||
|
||||
Complete diagram is Work-in-Progress.
|
||||
|
||||
### Notes
|
||||
|
||||
* Power ESP32 Mini with Li-Po battery using VCC (+) and GND (-) pins.
|
||||
* Connect the GY-91 board to the ESP32 Mini using VSPI, power it using 3.3V and GND pins:
|
||||
|
||||
|GY-91 pin|ESP32 pin|
|
||||
|-|-|
|
||||
|GND|GND|
|
||||
|3.3V|3.3V|
|
||||
|SCK|SVP (GPIO18)|
|
||||
|MOSI|GPIO23|
|
||||
|MISO|GPIO19|
|
||||
|NCS|GPIO5|
|
||||
|
||||
* Solder pull-down resistors to the MOSFETs.
|
||||
* Connect the motors to the ESP32 Mini using MOSFETs, by following scheme:
|
||||
|
||||
|Motor|Position|Direction|Wires|GPIO|
|
||||
|-|-|-|-|-|
|
||||
|Motor 0|Rear left|Counter-clockwise|Black & White|GPIO12|
|
||||
|Motor 1|Rear right|Clockwise|Blue & Red|GPIO13|
|
||||
|Motor 2|Front right|Counter-clockwise|Black & White|GPIO14|
|
||||
|Motor 3|Front left|Clockwise|Blue & Red|GPIO15|
|
||||
|
||||
Counter-clockwise motors have black and white wires and clockwise motors have blue and red wires.
|
||||
|
||||
* Optionally connect the RC receiver to the ESP32's UART2:
|
||||
|
||||
|Receiver pin|ESP32 pin|
|
||||
|-|-|
|
||||
|GND|GND|
|
||||
|VIN|VC (or 3.3V depending on the receiver)|
|
||||
|Signal|GPIO4⁴|
|
||||
|
||||
*⁴ — UART2 RX pin was [changed](https://docs.espressif.com/projects/arduino-esp32/en/latest/migration_guides/2.x_to_3.0.html#id14) to GPIO4 in Arduino ESP32 core 3.0.*
|
||||
|
||||
## Version 0
|
||||
|
||||
### Components
|
||||
See the information on the obsolete version 0 in the [corresponding article](docs/version0.md).
|
||||
|
||||
|Component|Type|Image|Quantity|
|
||||
|-|-|-|-|
|
||||
|ESP32 Mini|Microcontroller board|<img src="docs/img/esp32.jpg" width=100>|1|
|
||||
|GY-91|IMU+LDO+barometer board|<img src="docs/img/gy-91.jpg" width=100>|1|
|
||||
|K100|Quadcopter frame|<img src="docs/img/frame.jpg" width=100>|1|
|
||||
|8520 3.7V brushed motor|Motor|<img src="docs/img/motor.jpeg" width=100>|4|
|
||||
|Hubsan 55 mm| Propeller|<img src="docs/img/prop.jpg" width=100>|4|
|
||||
|2.7A 1S Dual Way Micro Brush ESC|Motor ESC|<img src="docs/img/esc.jpg" width=100>|4|
|
||||
|KINGKONG TINY X8|RC transmitter|<img src="docs/img/tx.jpg" width=100>|1|
|
||||
|DF500 (SBUS)|RC receiver|<img src="docs/img/rx.jpg" width=100>|1|
|
||||
||SBUS inverter|<img src="docs/img/inv.jpg" width=100>|1|
|
||||
|3.7 Li-Po 850 MaH 60C|Battery|||
|
||||
||Battery charger|<img src="docs/img/charger.jpg" width=100>|1|
|
||||
||Wires, connectors, tape, ...||
|
||||
||3D-printed frame parts||
|
||||
## Materials
|
||||
|
||||
Subscribe to the Telegram channel on developing the drone and the flight controller (in Russian): https://t.me/opensourcequadcopter.
|
||||
|
||||
Join the official Telegram chat: https://t.me/opensourcequadcopterchat.
|
||||
|
||||
Detailed article on Habr.com about the development of the drone (in Russian): https://habr.com/ru/articles/814127/.
|
||||
|
||||
@@ -1,5 +1,3 @@
|
||||
board_manager:
|
||||
additional_urls:
|
||||
- https://raw.githubusercontent.com/espressif/arduino-esp32/gh-pages/package_esp32_index.json
|
||||
library:
|
||||
enable_unsafe_install: true
|
||||
|
||||
BIN
docs/assets/esp32-holder.stl
Normal file
BIN
docs/assets/flix-frame.stl
Normal file
BIN
docs/assets/washer-m3.stl
Normal file
102
docs/build.md
@@ -1,12 +1,23 @@
|
||||
# Building and running
|
||||
|
||||
To build the firmware or the simulator, you need to clone the repository using git:
|
||||
|
||||
```bash
|
||||
git clone https://github.com/okalachev/flix.git
|
||||
cd flix
|
||||
```
|
||||
|
||||
## Simulation
|
||||
|
||||
Dependencies are [Gazebo Classic simulator](https://classic.gazebosim.org) and [SDL2](https://www.libsdl.org) library.
|
||||
|
||||
### Ubuntu
|
||||
|
||||
1. Install Gazebo 11:
|
||||
1. Install Arduino CLI:
|
||||
|
||||
```bash
|
||||
curl -fsSL https://raw.githubusercontent.com/arduino/arduino-cli/master/install.sh | BINDIR=~/.local/bin sh
|
||||
```
|
||||
|
||||
2. Install Gazebo 11:
|
||||
|
||||
```bash
|
||||
curl -sSL http://get.gazebosim.org | sh
|
||||
@@ -19,13 +30,19 @@ Dependencies are [Gazebo Classic simulator](https://classic.gazebosim.org) and [
|
||||
source ~/.bashrc
|
||||
```
|
||||
|
||||
2. Install SDL2:
|
||||
3. Install SDL2 and other dependencies:
|
||||
|
||||
```bash
|
||||
sudo apt-get install libsdl2-dev
|
||||
sudo apt-get update && sudo apt-get install build-essential libsdl2-dev
|
||||
```
|
||||
|
||||
3. Run the simulation:
|
||||
4. Add your user to the `input` group to enable joystick support (you need to re-login after this command):
|
||||
|
||||
```bash
|
||||
sudo usermod -a -G input $USER
|
||||
```
|
||||
|
||||
5. Run the simulation:
|
||||
|
||||
```bash
|
||||
make simulator
|
||||
@@ -39,27 +56,61 @@ Dependencies are [Gazebo Classic simulator](https://classic.gazebosim.org) and [
|
||||
/bin/bash -c "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/HEAD/install.sh)"
|
||||
```
|
||||
|
||||
2. Install Gazebo 11 and SDL2:
|
||||
2. Install Arduino CLI, Gazebo 11 and SDL2:
|
||||
|
||||
```bash
|
||||
brew tap osrf/simulation
|
||||
brew install arduino-cli
|
||||
brew install gazebo11
|
||||
brew install sdl2
|
||||
```
|
||||
|
||||
Set up your Gazebo environment variables:
|
||||
|
||||
```bash
|
||||
echo "source /opt/homebrew/share/gazebo/setup.sh" >> ~/.zshrc
|
||||
source ~/.zshrc
|
||||
```
|
||||
|
||||
3. Run the simulation:
|
||||
|
||||
```bash
|
||||
make simulator
|
||||
```
|
||||
|
||||
### Setup and flight
|
||||
|
||||
#### Control with smartphone
|
||||
|
||||
1. Install [QGroundControl mobile app](https://docs.qgroundcontrol.com/master/en/qgc-user-guide/getting_started/download_and_install.html#android) on your smartphone.
|
||||
2. Connect your smartphone to the same Wi-Fi network as the machine running the simulator.
|
||||
3. If you're using a virtual machine, make sure that its network is set to the **bridged** mode with Wi-Fi adapter selected.
|
||||
4. Run the simulation.
|
||||
5. Open QGroundControl app. It should connect and begin showing the virtual drone's telemetry automatically.
|
||||
6. Go to the settings and enable *Virtual Joystick*. *Auto-Center Throttle* setting **should be disabled**.
|
||||
7. Use the virtual joystick to fly the drone!
|
||||
|
||||
#### Control with USB remote control
|
||||
|
||||
1. Connect your USB remote control to the machine running the simulator.
|
||||
2. Run the simulation.
|
||||
3. Calibrate the RC using `cr` command in the command line interface and stop the simulation.
|
||||
4. Copy the calibration results to the source code (`gazebo/joystick.h`).
|
||||
5. Run the simulation again.
|
||||
6. Use the USB remote control to fly the drone!
|
||||
|
||||
## Firmware
|
||||
|
||||
### Arduino IDE (Windows, Linux, macOS)
|
||||
|
||||
1. Install [Arduino IDE](https://www.arduino.cc/en/software).
|
||||
2. Install ESP32 core using [Boards Manager](https://docs.arduino.cc/learn/starting-guide/cores).
|
||||
3. Build and upload the firmware using Arduino IDE.
|
||||
1. Install [Arduino IDE](https://www.arduino.cc/en/software) (version 2 is recommended).
|
||||
2. Install ESP32 core, version 3.0.3 (version 2.x is not supported). See the [official Espressif's instructions](https://docs.espressif.com/projects/arduino-esp32/en/latest/installing.html#installing-using-arduino-ide) on installing ESP32 Core in Arduino IDE.
|
||||
3. Install the following libraries using [Library Manager](https://docs.arduino.cc/software/ide-v2/tutorials/ide-v2-installing-a-library):
|
||||
* `FlixPeriph`.
|
||||
* `MAVLink`, version 2.0.10.
|
||||
4. Clone the project using git or [download the source code as a ZIP archive](https://codeload.github.com/okalachev/flix/zip/refs/heads/master).
|
||||
5. Open the downloaded Arduino sketch `flix/flix.ino` in Arduino IDE.
|
||||
6. [Build and upload](https://docs.arduino.cc/software/ide-v2/tutorials/getting-started/ide-v2-uploading-a-sketch) the firmware using Arduino IDE.
|
||||
|
||||
### Command line (Windows, Linux, macOS)
|
||||
|
||||
@@ -84,3 +135,34 @@ Dependencies are [Gazebo Classic simulator](https://classic.gazebosim.org) and [
|
||||
```
|
||||
|
||||
See other available Make commands in the [Makefile](../Makefile).
|
||||
|
||||
### Setup and flight
|
||||
|
||||
Before flight you need to calibrate the accelerometer:
|
||||
|
||||
1. Open Serial Monitor in Arduino IDE (use use `make monitor` command in the command line).
|
||||
2. Type `ca` command there.
|
||||
3. Copy calibration results to the source code (`flix/imu.ino`).
|
||||
|
||||
#### Control with smartphone
|
||||
|
||||
1. Install [QGroundControl mobile app](https://docs.qgroundcontrol.com/master/en/qgc-user-guide/getting_started/download_and_install.html#android) on your smartphone.
|
||||
2. Power the drone using the battery.
|
||||
3. Connect your smartphone to the appeared `flix` Wi-Fi network.
|
||||
4. Open QGroundControl app. It should connect and begin showing the drone's telemetry automatically.
|
||||
5. Go to the settings and enable *Virtual Joystick*. *Auto-Center Throttle* setting **should be disabled**.
|
||||
6. Use the virtual joystick to fly the drone!
|
||||
|
||||
#### Control with remote control
|
||||
|
||||
Before flight using remote control, you need to calibrate it:
|
||||
|
||||
1. Open Serial Monitor in Arduino IDE (use use `make monitor` command in the command line).
|
||||
2. Type `cr` command there.
|
||||
3. Copy calibration results to the source code (`flix/rc.ino`).
|
||||
|
||||
Then you can use your remote control to fly the drone!
|
||||
|
||||
### Firmware code structure
|
||||
|
||||
See [firmware overview](firmware.md) for more details.
|
||||
|
||||
37
docs/firmware.md
Normal file
@@ -0,0 +1,37 @@
|
||||
# Firmware overview
|
||||
|
||||
## Dataflow
|
||||
|
||||
<img src="img/dataflow.svg" width=800 alt="Firmware dataflow diagram">
|
||||
|
||||
The main loop is running at 1000 Hz. All the dataflow is happening through global variables (for simplicity):
|
||||
|
||||
* `t` *(float)* — current step time, *s*.
|
||||
* `dt` *(float)* — time delta between the current and previous steps, *s*.
|
||||
* `gyro` *(Vector)* — data from the gyroscope, *rad/s*.
|
||||
* `acc` *(Vector)* — acceleration data from the accelerometer, *m/s<sup>2</sup>*.
|
||||
* `rates` *(Vector)* — filtered angular rates, *rad/s*.
|
||||
* `attitude` *(Quaternion)* — estimated attitude (orientation) of drone.
|
||||
* `controls` *(float[])* — user control inputs from the RC, normalized to [-1, 1] range.
|
||||
* `motors` *(float[])* — motor outputs, normalized to [-1, 1] range; reverse rotation is possible.
|
||||
|
||||
## Source files
|
||||
|
||||
Firmware source files are located in `flix` directory. The key files are:
|
||||
|
||||
* [`flix.ino`](../flix/flix.ino) — main entry point, Arduino sketch. Includes global variables definition and the main loop.
|
||||
* [`imu.ino`](../flix/imu.ino) — reading data from the IMU sensor (gyroscope and accelerometer), IMU calibration.
|
||||
* [`rc.ino`](../flix/rc.ino) — reading data from the RC receiver, RC calibration.
|
||||
* [`estimate.ino`](../flix/estimate.ino) — drone's attitude estimation, complementary filter.
|
||||
* [`control.ino`](../flix/control.ino) — drone's attitude and rates control, three-dimensional two-level cascade PID controller.
|
||||
* [`motors.ino`](../flix/motors.ino) — PWM motor outputs control.
|
||||
|
||||
Utility files include:
|
||||
|
||||
* [`vector.h`](../flix/vector.h), [`quaternion.h`](../flix/quaternion.h) — project's vector and quaternion libraries implementation.
|
||||
* [`pid.h`](../flix/pid.h) — generic PID controller implementation.
|
||||
* [`lpf.h`](../flix/lpf.h) — generic low-pass filter implementation.
|
||||
|
||||
## Building
|
||||
|
||||
See build instructions in [build.md](build.md).
|
||||
BIN
docs/img/100n03a.jpg
Normal file
|
After Width: | Height: | Size: 27 KiB |
BIN
docs/img/battery.jpg
Normal file
|
After Width: | Height: | Size: 32 KiB |
330
docs/img/dataflow.svg
Normal file
@@ -0,0 +1,330 @@
|
||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||
<!-- Created with Inkscape (http://www.inkscape.org/) -->
|
||||
|
||||
<svg
|
||||
width="613.59802mm"
|
||||
height="267.24701mm"
|
||||
viewBox="0 -10 613.59802 267.247"
|
||||
version="1.1"
|
||||
id="svg1"
|
||||
xml:space="preserve"
|
||||
inkscape:version="1.3.2 (091e20e, 2023-11-25)"
|
||||
sodipodi:docname="dataflow.svg"
|
||||
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||
xmlns:xlink="http://www.w3.org/1999/xlink"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
xmlns:svg="http://www.w3.org/2000/svg"><sodipodi:namedview
|
||||
id="namedview1"
|
||||
pagecolor="#ffffff"
|
||||
bordercolor="#666666"
|
||||
borderopacity="1.0"
|
||||
inkscape:showpageshadow="2"
|
||||
inkscape:pageopacity="0.0"
|
||||
inkscape:pagecheckerboard="0"
|
||||
inkscape:deskcolor="#d1d1d1"
|
||||
inkscape:document-units="mm"
|
||||
inkscape:zoom="0.34862039"
|
||||
inkscape:cx="1219.091"
|
||||
inkscape:cy="608.1113"
|
||||
inkscape:window-width="1496"
|
||||
inkscape:window-height="905"
|
||||
inkscape:window-x="0"
|
||||
inkscape:window-y="34"
|
||||
inkscape:window-maximized="1"
|
||||
inkscape:current-layer="svg1"><inkscape:page
|
||||
x="0"
|
||||
y="0"
|
||||
width="613.59802"
|
||||
height="267.24701"
|
||||
id="page2"
|
||||
margin="0"
|
||||
bleed="0" /><inkscape:page
|
||||
x="-30.32262"
|
||||
y="-66.876167"
|
||||
width="677.33331"
|
||||
height="381"
|
||||
id="page3"
|
||||
margin="0"
|
||||
bleed="0" /></sodipodi:namedview><defs
|
||||
id="defs1"><color-profile
|
||||
inkscape:label="sRGB IEC61966-2.1"
|
||||
name="sRGB-IEC61966-2.1"
|
||||
xlink:href="data:application/vnd.iccprofile;base64,AAAMbGxjbXMCEAAAbW50clJHQiBYWVogB84AAgAJAAYAMQAAYWNzcEFQUEwAAAAASUVDIHNSR0IAAAAAAAAAAAAAAAAAAPbWAAEAAAAA0y1sY21zAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAARY3BydAAAAVAAAAAzZGVzYwAAAYQAAACQd3RwdAAAAhQAAAAUYmtwdAAAAigAAAAUclhZWgAAAjwAAAAUZ1hZWgAAAlAAAAAUYlhZWgAAAmQAAAAUZG1uZAAAAngAAABwZG1kZAAAAugAAACIdnVlZAAAA3AAAACGdmlldwAAA/gAAAAkbHVtaQAABBwAAAAUbWVhcwAABDAAAAAkdGVjaAAABFQAAAAMclRSQwAABGAAAAgMZ1RSQwAABGAAAAgMYlRSQwAABGAAAAgMdGV4dAAAAABDb3B5cmlnaHQgKGMpIDE5OTggSGV3bGV0dC1QYWNrYXJkIENvbXBhbnkAAGRlc2MAAAAAAAAAEnNSR0IgSUVDNjE5NjYtMi4xAAAAAAAAAAASAHMAUgBHAEIAIABJAEUAQwA2ADEAOQA2ADYALQAyAC4AMQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAFhZWiAAAAAAAADzUQABAAAAARbMWFlaIAAAAAAAAAAAAAAAAAAAAABYWVogAAAAAAAAb6IAADj1AAADkFhZWiAAAAAAAABimQAAt4UAABjaWFlaIAAAAAAAACSgAAAPhAAAts9kZXNjAAAAAAAAABZJRUMgaHR0cDovL3d3dy5pZWMuY2gAAAAAAAAAAAAAABZJRUMgaHR0cDovL3d3dy5pZWMuY2gAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAZGVzYwAAAAAAAAAuSUVDIDYxOTY2LTIuMSBEZWZhdWx0IFJHQiBjb2xvdXIgc3BhY2UgLSBzUkdCAAAAAAAAAAAAAAAuSUVDIDYxOTY2LTIuMSBEZWZhdWx0IFJHQiBjb2xvdXIgc3BhY2UgLSBzUkdCAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGRlc2MAAAAAAAAALFJlZmVyZW5jZSBWaWV3aW5nIENvbmRpdGlvbiBpbiBJRUM2MTk2Ni0yLjEAAAAAAAAAAAAAACxSZWZlcmVuY2UgVmlld2luZyBDb25kaXRpb24gaW4gSUVDNjE5NjYtMi4xAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAB2aWV3AAAAAAATpP4AFF8uABDPFAAD7cwABBMLAANcngAAAAFYWVogAAAAAABMCVYAUAAAAFcf521lYXMAAAAAAAAAAQAAAAAAAAAAAAAAAAAAAAAAAAKPAAAAAnNpZyAAAAAAQ1JUIGN1cnYAAAAAAAAEAAAAAAUACgAPABQAGQAeACMAKAAtADIANwA7AEAARQBKAE8AVABZAF4AYwBoAG0AcgB3AHwAgQCGAIsAkACVAJoAnwCkAKkArgCyALcAvADBAMYAywDQANUA2wDgAOUA6wDwAPYA+wEBAQcBDQETARkBHwElASsBMgE4AT4BRQFMAVIBWQFgAWcBbgF1AXwBgwGLAZIBmgGhAakBsQG5AcEByQHRAdkB4QHpAfIB+gIDAgwCFAIdAiYCLwI4AkECSwJUAl0CZwJxAnoChAKOApgCogKsArYCwQLLAtUC4ALrAvUDAAMLAxYDIQMtAzgDQwNPA1oDZgNyA34DigOWA6IDrgO6A8cD0wPgA+wD+QQGBBMEIAQtBDsESARVBGMEcQR+BIwEmgSoBLYExATTBOEE8AT+BQ0FHAUrBToFSQVYBWcFdwWGBZYFpgW1BcUF1QXlBfYGBgYWBicGNwZIBlkGagZ7BowGnQavBsAG0QbjBvUHBwcZBysHPQdPB2EHdAeGB5kHrAe/B9IH5Qf4CAsIHwgyCEYIWghuCIIIlgiqCL4I0gjnCPsJEAklCToJTwlkCXkJjwmkCboJzwnlCfsKEQonCj0KVApqCoEKmAquCsUK3ArzCwsLIgs5C1ELaQuAC5gLsAvIC+EL+QwSDCoMQwxcDHUMjgynDMAM2QzzDQ0NJg1ADVoNdA2ODakNww3eDfgOEw4uDkkOZA5/DpsOtg7SDu4PCQ8lD0EPXg96D5YPsw/PD+wQCRAmEEMQYRB+EJsQuRDXEPURExExEU8RbRGMEaoRyRHoEgcSJhJFEmQShBKjEsMS4xMDEyMTQxNjE4MTpBPFE+UUBhQnFEkUahSLFK0UzhTwFRIVNBVWFXgVmxW9FeAWAxYmFkkWbBaPFrIW1hb6Fx0XQRdlF4kXrhfSF/cYGxhAGGUYihivGNUY+hkgGUUZaxmRGbcZ3RoEGioaURp3Gp4axRrsGxQbOxtjG4obshvaHAIcKhxSHHscoxzMHPUdHh1HHXAdmR3DHeweFh5AHmoelB6+HukfEx8+H2kflB+/H+ogFSBBIGwgmCDEIPAhHCFIIXUhoSHOIfsiJyJVIoIiryLdIwojOCNmI5QjwiPwJB8kTSR8JKsk2iUJJTglaCWXJccl9yYnJlcmhya3JugnGCdJJ3onqyfcKA0oPyhxKKIo1CkGKTgpaymdKdAqAio1KmgqmyrPKwIrNitpK50r0SwFLDksbiyiLNctDC1BLXYtqy3hLhYuTC6CLrcu7i8kL1ovkS/HL/4wNTBsMKQw2zESMUoxgjG6MfIyKjJjMpsy1DMNM0YzfzO4M/E0KzRlNJ402DUTNU01hzXCNf02NzZyNq426TckN2A3nDfXOBQ4UDiMOMg5BTlCOX85vDn5OjY6dDqyOu87LTtrO6o76DwnPGU8pDzjPSI9YT2hPeA+ID5gPqA+4D8hP2E/oj/iQCNAZECmQOdBKUFqQaxB7kIwQnJCtUL3QzpDfUPARANER0SKRM5FEkVVRZpF3kYiRmdGq0bwRzVHe0fASAVIS0iRSNdJHUljSalJ8Eo3Sn1KxEsMS1NLmkviTCpMcky6TQJNSk2TTdxOJU5uTrdPAE9JT5NP3VAnUHFQu1EGUVBRm1HmUjFSfFLHUxNTX1OqU/ZUQlSPVNtVKFV1VcJWD1ZcVqlW91dEV5JX4FgvWH1Yy1kaWWlZuFoHWlZaplr1W0VblVvlXDVchlzWXSddeF3JXhpebF69Xw9fYV+zYAVgV2CqYPxhT2GiYfViSWKcYvBjQ2OXY+tkQGSUZOllPWWSZedmPWaSZuhnPWeTZ+loP2iWaOxpQ2maafFqSGqfavdrT2una/9sV2yvbQhtYG25bhJua27Ebx5veG/RcCtwhnDgcTpxlXHwcktypnMBc11zuHQUdHB0zHUodYV14XY+dpt2+HdWd7N4EXhueMx5KnmJeed6RnqlewR7Y3vCfCF8gXzhfUF9oX4BfmJ+wn8jf4R/5YBHgKiBCoFrgc2CMIKSgvSDV4O6hB2EgITjhUeFq4YOhnKG14c7h5+IBIhpiM6JM4mZif6KZIrKizCLlov8jGOMyo0xjZiN/45mjs6PNo+ekAaQbpDWkT+RqJIRknqS45NNk7aUIJSKlPSVX5XJljSWn5cKl3WX4JhMmLiZJJmQmfyaaJrVm0Kbr5wcnImc951kndKeQJ6unx2fi5/6oGmg2KFHobaiJqKWowajdqPmpFakx6U4pammGqaLpv2nbqfgqFKoxKk3qamqHKqPqwKrdavprFys0K1ErbiuLa6hrxavi7AAsHWw6rFgsdayS7LCszizrrQltJy1E7WKtgG2ebbwt2i34LhZuNG5SrnCuju6tbsuu6e8IbybvRW9j74KvoS+/796v/XAcMDswWfB48JfwtvDWMPUxFHEzsVLxcjGRsbDx0HHv8g9yLzJOsm5yjjKt8s2y7bMNcy1zTXNtc42zrbPN8+40DnQutE80b7SP9LB00TTxtRJ1MvVTtXR1lXW2Ndc1+DYZNjo2WzZ8dp22vvbgNwF3IrdEN2W3hzeot8p36/gNuC94UThzOJT4tvjY+Pr5HPk/OWE5g3mlucf56noMui86Ubp0Opb6uXrcOv77IbtEe2c7ijutO9A78zwWPDl8XLx//KM8xnzp/Q09ML1UPXe9m32+/eK+Bn4qPk4+cf6V/rn+3f8B/yY/Sn9uv5L/tz/bf//"
|
||||
id="color-profile1" /><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath2"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
id="path2" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath4"><path
|
||||
d="m 745.9206,375.0079 h 404.1921 V 493.7148 H 745.9206 Z"
|
||||
transform="matrix(1,0,0,-1,-874.5547,410.75391)"
|
||||
id="path4" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath6"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
id="path6" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath7"><path
|
||||
d="m 734.9959,651.142 h 426.0414 V 769.8489 H 734.9959 Z"
|
||||
transform="matrix(1,0,0,-1,-786.22464,686.88802)"
|
||||
id="path7" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath9"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
id="path9" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath10"><path
|
||||
d="m 67.84839,771.7229 h 404.1921 V 890.4298 H 67.84839 Z"
|
||||
transform="matrix(1,0,0,-1,-172.21011,807.46902)"
|
||||
id="path10" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath12"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
id="path12" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath13"><path
|
||||
d="m 1452.804,771.7229 h 375.3804 V 890.4298 H 1452.804 Z"
|
||||
transform="matrix(1,0,0,-1,-1500.8361,807.46902)"
|
||||
id="path13" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath15"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
id="path15" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath16"><path
|
||||
d="m 1452.804,226.7937 h 375.3804 V 345.5006 H 1452.804 Z"
|
||||
transform="matrix(1,0,0,-1,-1499.533,262.53983)"
|
||||
id="path16" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath18"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
transform="matrix(1,0,0,-1,-451.65591,778.64052)"
|
||||
id="path18" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath20"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
id="path20" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath21"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
transform="matrix(0.98150515,-0.1914358,-0.1914358,-0.98150515,-353.6359,868.10017)"
|
||||
id="path21" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath23"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
transform="matrix(0.89875521,0.43845065,0.43845065,-0.89875521,-1406.8193,-156.93149)"
|
||||
id="path23" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath25"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
transform="matrix(0.98480768,0.17364817,0.17364817,-0.98480768,-1293.1181,526.43111)"
|
||||
id="path25" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath27"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
transform="matrix(1,0,0,-1,-1132.0071,771.72292)"
|
||||
id="path27" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath30"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
transform="matrix(1,0,0,-1,-1142.931,782.32692)"
|
||||
id="path30" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath33"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
transform="matrix(1,0,0,-1,-1656.51,771.72572)"
|
||||
id="path33" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath35"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
transform="matrix(1,0,0,-1,-1703.37,542.87351)"
|
||||
id="path35" /></clipPath><clipPath
|
||||
clipPathUnits="userSpaceOnUse"
|
||||
id="clipPath37"><path
|
||||
d="M 0,0 H 1920 V 1080 H 0 Z"
|
||||
transform="matrix(1,0,0,-1,-270.40501,771.68402)"
|
||||
id="path37" /></clipPath></defs><g
|
||||
inkscape:label="Layer 1"
|
||||
inkscape:groupmode="layer"
|
||||
id="layer1"
|
||||
transform="translate(-250.32262,-66.876165)" /><g
|
||||
id="g1"
|
||||
inkscape:groupmode="layer"
|
||||
inkscape:label="1"
|
||||
transform="matrix(0.26458333,0,0,0.26458334,-30.322612,-66.876165)"><g
|
||||
id="g2"><path
|
||||
id="path3"
|
||||
d="m 786.6761,493.7148 h 322.6809 c 6.647,0 10.635,0 13.293,-1.11 3.833,-1.3949 6.852,-4.4141 8.247,-8.2468 1.11,-2.6586 1.11,-6.6466 1.11,-13.2932 v -73.4069 c 0,-6.6466 0,-10.6346 -1.11,-13.2932 -1.395,-3.8327 -4.414,-6.8518 -8.247,-8.2468 -2.658,-1.11 -6.646,-1.11 -13.293,-1.11 H 786.6761 c -6.6467,0 -10.6346,0 -13.2933,1.11 -3.8327,1.395 -6.8518,4.4141 -8.2468,8.2468 -1.1099,2.6586 -1.1099,6.6466 -1.1099,13.2932 v 73.4069 c 0,6.6466 0,10.6346 1.1099,13.2932 1.395,3.8327 4.4141,6.8519 8.2468,8.2468 2.6587,1.11 6.6466,1.11 13.2933,1.11 z"
|
||||
style="fill:#0076ba;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
transform="matrix(1.3333333,0,0,-1.3333333,0,1440)" /></g><g
|
||||
id="g3"><text
|
||||
id="text3"
|
||||
xml:space="preserve"
|
||||
transform="matrix(1.3333333,0,0,1.3333333,1166.0729,892.32813)"
|
||||
clip-path="url(#clipPath4)"><tspan
|
||||
style="font-variant:normal;font-weight:normal;font-size:60px;font-family:Tahoma;writing-mode:lr-tb;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
x="0 21.33 49.02 67.169998 80.879997 114.33"
|
||||
y="0"
|
||||
sodipodi:role="line"
|
||||
id="tspan3">rc.ino</tspan></text></g><g
|
||||
id="g4"><path
|
||||
id="path5"
|
||||
d="m 775.7514,769.8489 h 344.5306 c 6.646,0 10.634,0 13.293,-1.11 3.833,-1.395 6.852,-4.4141 8.247,-8.2468 1.11,-2.6586 1.11,-6.6466 1.11,-13.2932 V 673.792 c 0,-6.6466 0,-10.6346 -1.11,-13.2932 -1.395,-3.8327 -4.414,-6.8519 -8.247,-8.2468 -2.659,-1.11 -6.647,-1.11 -13.293,-1.11 H 775.7514 c -6.6466,0 -10.6346,0 -13.2933,1.11 -3.8326,1.3949 -6.8518,4.4141 -8.2468,8.2468 -1.1099,2.6586 -1.1099,6.6466 -1.1099,13.2932 v 73.4069 c 0,6.6466 0,10.6346 1.1099,13.2932 1.395,3.8327 4.4142,6.8518 8.2468,8.2468 2.6587,1.11 6.6467,1.11 13.2933,1.11 z"
|
||||
style="fill:#0076ba;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
transform="matrix(1.3333333,0,0,-1.3333333,0,1440)"
|
||||
clip-path="url(#clipPath6)" /></g><g
|
||||
id="g6"><text
|
||||
id="text6"
|
||||
xml:space="preserve"
|
||||
transform="matrix(1.3333333,0,0,1.3333333,1048.2995,524.14933)"
|
||||
clip-path="url(#clipPath7)"><tspan
|
||||
style="font-variant:normal;font-weight:normal;font-size:60px;font-family:Tahoma;writing-mode:lr-tb;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
x="0 31.584 58.368 78.431999 92.136002 142.5 173.964 194.028 225.612 243.756 257.45999 290.90399"
|
||||
y="0"
|
||||
sodipodi:role="line"
|
||||
id="tspan6">estimate.ino</tspan></text></g><g
|
||||
id="g7"><path
|
||||
id="path8"
|
||||
d="M 108.6039,890.4298 H 431.285 c 6.6466,0 10.6346,0 13.2932,-1.1099 3.8327,-1.395 6.8519,-4.4142 8.2468,-8.2468 1.11,-2.6587 1.11,-6.6467 1.11,-13.2933 V 794.373 c 0,-6.6467 0,-10.6346 -1.11,-13.2933 -1.3949,-3.8327 -4.4141,-6.8518 -8.2468,-8.2468 -2.6586,-1.11 -6.6466,-1.11 -13.2932,-1.11 H 108.6039 c -6.6467,0 -10.63463,0 -13.29329,1.11 -3.83267,1.395 -6.85182,4.4141 -8.2468,8.2468 -1.10995,2.6587 -1.10995,6.6466 -1.10995,13.2933 v 73.4068 c 0,6.6466 0,10.6346 1.10995,13.2933 1.39498,3.8326 4.41413,6.8518 8.2468,8.2468 2.65866,1.1099 6.64659,1.1099 13.29329,1.1099 z"
|
||||
style="fill:#0076ba;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
transform="matrix(1.3333333,0,0,-1.3333333,0,1440)"
|
||||
clip-path="url(#clipPath9)" /></g><g
|
||||
id="g9"><text
|
||||
id="text9"
|
||||
xml:space="preserve"
|
||||
transform="matrix(1.3333333,0,0,1.3333333,229.61347,363.37467)"
|
||||
clip-path="url(#clipPath10)"><tspan
|
||||
style="font-variant:normal;font-weight:normal;font-size:60px;font-family:Tahoma;writing-mode:lr-tb;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
x="0 13.71 64.080002 97.529999 115.68 129.39 162.84"
|
||||
y="0"
|
||||
sodipodi:role="line"
|
||||
id="tspan9">imu.ino</tspan></text></g><g
|
||||
id="g10"><path
|
||||
id="path11"
|
||||
d="m 1493.56,890.4298 h 293.869 c 6.647,0 10.635,0 13.294,-1.1099 3.832,-1.395 6.851,-4.4142 8.246,-8.2468 1.11,-2.6587 1.11,-6.6467 1.11,-13.2933 V 794.373 c 0,-6.6467 0,-10.6346 -1.11,-13.2933 -1.395,-3.8327 -4.414,-6.8518 -8.246,-8.2468 -2.659,-1.11 -6.647,-1.11 -13.294,-1.11 H 1493.56 c -6.647,0 -10.635,0 -13.293,1.11 -3.833,1.395 -6.852,4.4141 -8.247,8.2468 -1.11,2.6587 -1.11,6.6466 -1.11,13.2933 v 73.4068 c 0,6.6466 0,10.6346 1.11,13.2933 1.395,3.8326 4.414,6.8518 8.247,8.2468 2.658,1.1099 6.646,1.1099 13.293,1.1099 z"
|
||||
style="fill:#0076ba;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
transform="matrix(1.3333333,0,0,-1.3333333,0,1440)"
|
||||
clip-path="url(#clipPath12)" /></g><g
|
||||
id="g12"><text
|
||||
id="text12"
|
||||
xml:space="preserve"
|
||||
transform="matrix(1.3333333,0,0,1.3333333,2001.1147,363.37467)"
|
||||
clip-path="url(#clipPath13)"><tspan
|
||||
style="font-variant:normal;font-weight:normal;font-size:60px;font-family:Tahoma;writing-mode:lr-tb;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
x="0 27.684 60.264 93.720001 113.79 135.12 167.7 181.41 199.57201 213.282 246.73801"
|
||||
y="0"
|
||||
sodipodi:role="line"
|
||||
id="tspan12">control.ino</tspan></text></g><g
|
||||
id="g13"><path
|
||||
id="path14"
|
||||
d="m 1493.56,345.5006 h 293.869 c 6.647,0 10.635,0 13.294,-1.1099 3.832,-1.395 6.851,-4.4142 8.246,-8.2468 1.11,-2.6587 1.11,-6.6467 1.11,-13.2933 v -73.4068 c 0,-6.6467 0,-10.6346 -1.11,-13.2933 -1.395,-3.8327 -4.414,-6.8518 -8.246,-8.2468 -2.659,-1.11 -6.647,-1.11 -13.294,-1.11 H 1493.56 c -6.647,0 -10.635,0 -13.293,1.11 -3.833,1.395 -6.852,4.4141 -8.247,8.2468 -1.11,2.6587 -1.11,6.6466 -1.11,13.2933 v 73.4068 c 0,6.6466 0,10.6346 1.11,13.2933 1.395,3.8326 4.414,6.8518 8.247,8.2468 2.658,1.1099 6.646,1.1099 13.293,1.1099 z"
|
||||
style="fill:#0076ba;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
transform="matrix(1.3333333,0,0,-1.3333333,0,1440)"
|
||||
clip-path="url(#clipPath15)" /></g><g
|
||||
id="g15"><text
|
||||
id="text15"
|
||||
xml:space="preserve"
|
||||
transform="matrix(1.3333333,0,0,1.3333333,1999.3773,1089.9469)"
|
||||
clip-path="url(#clipPath16)"><tspan
|
||||
style="font-variant:normal;font-weight:normal;font-size:60px;font-family:Tahoma;writing-mode:lr-tb;fill:#ffffff;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
x="0 50.388 82.968002 103.038 135.618 157.242 184.02 202.18201 215.892 249.34801"
|
||||
y="0"
|
||||
sodipodi:role="line"
|
||||
id="tspan15">motors.ino</tspan></text></g><g
|
||||
id="g16"><path
|
||||
id="path17"
|
||||
d="m 0,0 c 89.24774,21.64997 181.7839,38.35706 277.6084,50.12129 l 2.9802,0.35217"
|
||||
style="fill:none;stroke:#d5d5d5;stroke-width:6;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
||||
transform="matrix(1.3333333,0,0,1.3333333,602.20787,401.81267)"
|
||||
clip-path="url(#clipPath18)" /><path
|
||||
id="path19"
|
||||
d="m 727.8571,716.602 25.2424,9.1006 -22.426,14.7336 z"
|
||||
style="fill:#d5d5d5;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
transform="matrix(1.3333333,0,0,-1.3333333,0,1440)"
|
||||
clip-path="url(#clipPath20)" /></g><g
|
||||
id="g20" /><g
|
||||
id="g21"><text
|
||||
id="text21"
|
||||
xml:space="preserve"
|
||||
transform="matrix(1.3086735,0.25524773,-0.25524773,1.3086735,684.37453,394.20507)"
|
||||
clip-path="url(#clipPath21)"><tspan
|
||||
style="font-variant:normal;font-weight:normal;font-size:40px;font-family:Tahoma;writing-mode:lr-tb;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
x="0 22.108 42.028 56.248001 77.400002 89.508003 102.008 123.004 141.34399"
|
||||
y="0"
|
||||
sodipodi:role="line"
|
||||
id="tspan21">gyro, acc</tspan></text></g><g
|
||||
id="g22" /><g
|
||||
id="g23"><text
|
||||
id="text23"
|
||||
xml:space="preserve"
|
||||
transform="matrix(1.1983404,-0.58460093,0.58460093,1.1983404,1777.5907,805.62947)"
|
||||
clip-path="url(#clipPath23)"><tspan
|
||||
style="font-variant:normal;font-weight:normal;font-size:40px;font-family:Tahoma;writing-mode:lr-tb;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
x="0 18.455999 40.175999 62.48 75.860001 90.080002 111.8 120.94 138.79201 154.104 175.94 197.776"
|
||||
y="0"
|
||||
sodipodi:role="line"
|
||||
id="tspan23">controls[16]</tspan></text></g><g
|
||||
id="g24" /><g
|
||||
id="g25"><text
|
||||
id="text25"
|
||||
xml:space="preserve"
|
||||
transform="matrix(1.3130771,-0.23153093,0.23153093,1.3130771,1576.0787,449.35853)"
|
||||
clip-path="url(#clipPath25)"><tspan
|
||||
style="font-variant:normal;font-weight:normal;font-size:40px;font-family:Tahoma;writing-mode:lr-tb;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
x="0 13.732 34.743999 48.116001 69.167999 87.019997 99.112 111.604 132.616 145.668 159.03999 168.252 181.62399 203.916 226.008"
|
||||
y="0"
|
||||
sodipodi:role="line"
|
||||
id="tspan25">rates, attitude</tspan></text></g><g
|
||||
id="g26"><path
|
||||
id="path26"
|
||||
d="M 0,306.3767 C 237.7238,253.1408 395.5189,158.083 473.3852,21.20323 l 1.4097,-2.65533"
|
||||
style="fill:none;stroke:#d5d5d5;stroke-width:6;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
||||
transform="matrix(1.3333333,0,0,1.3333333,1509.3427,411.03613)"
|
||||
clip-path="url(#clipPath27)" /><path
|
||||
id="path28"
|
||||
d="m 1615.994,744.8981 0.656,26.8248 -21.853,-15.5703 z"
|
||||
style="fill:#d5d5d5;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
transform="matrix(1.3333333,0,0,-1.3333333,0,1440)" /></g><g
|
||||
id="g28"><path
|
||||
id="path29"
|
||||
d="M 0,60.29815 C 99.1961,49.5746 200.9904,31.41576 305.383,5.821648 l 2.914,-0.728473"
|
||||
style="fill:none;stroke:#d5d5d5;stroke-width:6;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
||||
transform="matrix(1.3333333,0,0,1.3333333,1523.908,396.89747)"
|
||||
clip-path="url(#clipPath30)" /><path
|
||||
id="path31"
|
||||
d="m 1451.228,764.8644 20.373,17.4625 -26.194,5.821 z"
|
||||
style="fill:#d5d5d5;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
transform="matrix(1.3333333,0,0,-1.3333333,0,1440)" /></g><g
|
||||
id="g31"><path
|
||||
id="path32"
|
||||
d="M 5.061111,0 C 39.5121,116.8397 39.6576,251.1261 5.497595,402.859 l -0.68786,2.9234"
|
||||
style="fill:none;stroke:#d5d5d5;stroke-width:6;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
|
||||
transform="matrix(1.3333333,0,0,1.3333333,2208.68,411.0324)"
|
||||
clip-path="url(#clipPath33)" /><path
|
||||
id="path34"
|
||||
d="m 1650.326,371.6119 6.184,-26.1104 17.178,20.6136 z"
|
||||
style="fill:#d5d5d5;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
transform="matrix(1.3333333,0,0,-1.3333333,0,1440)" /></g><g
|
||||
id="g34" /><g
|
||||
id="g35"><text
|
||||
id="text35"
|
||||
xml:space="preserve"
|
||||
transform="matrix(1.3333333,0,0,1.3333333,2271.16,716.16867)"
|
||||
clip-path="url(#clipPath35)"><tspan
|
||||
style="font-variant:normal;font-weight:normal;font-size:40px;font-family:Tahoma;writing-mode:lr-tb;fill:#000000;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
x="0 33.591999 55.312 68.692001 90.412003 104.828"
|
||||
y="0"
|
||||
sodipodi:role="line"
|
||||
id="tspan35">motors</tspan></text></g><g
|
||||
id="g36"><path
|
||||
id="path36"
|
||||
d="M 1200.4,523.3549 C 431.1994,677.8194 31.45986,511.3581 1.181784,23.97107 L 1.034053,20.97453"
|
||||
style="fill:none;stroke:#ff9300;stroke-width:6;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:12, 12;stroke-dashoffset:0;stroke-opacity:1"
|
||||
transform="matrix(1.3333333,0,0,1.3333333,360.54,411.088)"
|
||||
clip-path="url(#clipPath37)" /><path
|
||||
id="path38"
|
||||
d="m 283.5722,748.304 -13.1672,23.38 -10.8036,-24.5618 z"
|
||||
style="fill:#ff9300;fill-opacity:1;fill-rule:nonzero;stroke:none"
|
||||
transform="matrix(1.3333333,0,0,-1.3333333,0,1440)" /></g></g></svg>
|
||||
|
After Width: | Height: | Size: 22 KiB |
BIN
docs/img/esp32-holder.jpg
Normal file
|
After Width: | Height: | Size: 9.3 KiB |
BIN
docs/img/flight-video.jpg
Normal file
|
After Width: | Height: | Size: 79 KiB |
BIN
docs/img/flightplot.png
Normal file
|
After Width: | Height: | Size: 56 KiB |
BIN
docs/img/flix1.jpg
Normal file
|
After Width: | Height: | Size: 89 KiB |
BIN
docs/img/foxglove.png
Normal file
|
After Width: | Height: | Size: 61 KiB |
BIN
docs/img/frame1.jpg
Normal file
|
After Width: | Height: | Size: 11 KiB |
BIN
docs/img/mosfet-connection.png
Normal file
|
After Width: | Height: | Size: 38 KiB |
BIN
docs/img/plotjuggler.png
Normal file
|
After Width: | Height: | Size: 53 KiB |
BIN
docs/img/resistor10k.jpg
Normal file
|
After Width: | Height: | Size: 12 KiB |
|
Before Width: | Height: | Size: 37 KiB After Width: | Height: | Size: 42 KiB |
256
docs/img/schematics1.svg
Normal file
@@ -0,0 +1,256 @@
|
||||
<?xml version="1.0" encoding="utf-8"?>
|
||||
<svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
|
||||
viewBox="0 0 1920 1080" style="enable-background:new 0 0 1920 1080;" xml:space="preserve">
|
||||
<style type="text/css">
|
||||
.st0{clip-path:url(#SVGID_00000116955662310502408250000008996271717606231736_);fill:#FFFFFF;}
|
||||
.st1{clip-path:url(#SVGID_00000116955662310502408250000008996271717606231736_);fill:#0076BA;}
|
||||
.st2{clip-path:url(#SVGID_00000116955662310502408250000008996271717606231736_);fill:none;stroke:#0076BA;stroke-width:6;}
|
||||
.st3{clip-path:url(#SVGID_00000055674346191406539380000013421132283630177205_);}
|
||||
.st4{fill:#FFFFFF;}
|
||||
.st5{font-family:'Tahoma';}
|
||||
.st6{font-size:60px;}
|
||||
.st7{clip-path:url(#SVGID_00000057846011469822040540000011754501750068092081_);fill:none;stroke:#0076BA;stroke-width:6;}
|
||||
|
||||
.st8{clip-path:url(#SVGID_00000057846011469822040540000011754501750068092081_);fill:none;stroke:#0076BA;stroke-width:6;stroke-dasharray:12,12;}
|
||||
.st9{clip-path:url(#SVGID_00000031912132892401345140000018376817810309323944_);}
|
||||
.st10{letter-spacing:-1;}
|
||||
.st11{clip-path:url(#SVGID_00000096022444481931167570000018189382621257791423_);fill:none;stroke:#D5D5D5;stroke-width:6;}
|
||||
.st12{clip-path:url(#SVGID_00000096022444481931167570000018189382621257791423_);fill:#D5D5D5;}
|
||||
.st13{clip-path:url(#SVGID_00000096022444481931167570000018189382621257791423_);}
|
||||
.st14{font-size:40px;}
|
||||
|
||||
.st15{clip-path:url(#SVGID_00000096022444481931167570000018189382621257791423_);fill:none;stroke:#D5D5D5;stroke-width:6;stroke-dasharray:12,12;}
|
||||
.st16{letter-spacing:-3;}
|
||||
.st17{clip-path:url(#SVGID_00000096022444481931167570000018189382621257791423_);fill:none;stroke:#0076BA;stroke-width:6;}
|
||||
.st18{clip-path:url(#SVGID_00000178923025368094801390000008109591059568692644_);}
|
||||
.st19{clip-path:url(#SVGID_00000170245726615134129150000014803823133490835355_);fill:none;stroke:#D5D5D5;stroke-width:6;}
|
||||
.st20{clip-path:url(#SVGID_00000170245726615134129150000014803823133490835355_);fill:#D5D5D5;}
|
||||
.st21{clip-path:url(#SVGID_00000170245726615134129150000014803823133490835355_);}
|
||||
.st22{clip-path:url(#SVGID_00000170245726615134129150000014803823133490835355_);fill:none;stroke:#FF9300;stroke-width:6;}
|
||||
.st23{clip-path:url(#SVGID_00000137829079020238483810000004945639728221863820_);}
|
||||
.st24{clip-path:url(#SVGID_00000127742992435002359440000017943924755103710901_);fill:none;stroke:#0076BA;stroke-width:6;}
|
||||
.st25{clip-path:url(#SVGID_00000172413095142863711730000006643673940354901182_);}
|
||||
.st26{fill:#333333;}
|
||||
.st27{clip-path:url(#SVGID_00000057841300642942441540000015331613196773382808_);fill:none;stroke:#0076BA;stroke-width:3;}
|
||||
.st28{clip-path:url(#SVGID_00000015346235126698654330000007941041458912523396_);}
|
||||
.st29{font-size:30px;}
|
||||
</style>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_1_" width="1920" height="1080"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000171677823290121104880000004951624621648774806_">
|
||||
<use xlink:href="#SVGID_1_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
|
||||
<rect style="clip-path:url(#SVGID_00000171677823290121104880000004951624621648774806_);fill:#FFFFFF;" width="1920" height="1080"/>
|
||||
<path style="clip-path:url(#SVGID_00000171677823290121104880000004951624621648774806_);fill:#0076BA;" d="M781,353.4h358
|
||||
c8.4,0,13.4,0,16.7,1.4c4.8,1.8,8.6,5.6,10.4,10.4c1.4,3.3,1.4,8.4,1.4,16.7v268.8c0,8.4,0,13.4-1.4,16.7
|
||||
c-1.8,4.8-5.6,8.6-10.4,10.4c-3.3,1.4-8.4,1.4-16.7,1.4H781c-8.4,0-13.4,0-16.7-1.4c-4.8-1.8-8.6-5.6-10.4-10.4
|
||||
c-1.4-3.3-1.4-8.4-1.4-16.7V381.9c0-8.4,0-13.4,1.4-16.7c1.8-4.8,5.6-8.6,10.4-10.4C767.6,353.4,772.6,353.4,781,353.4z"/>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000171677823290121104880000004951624621648774806_);fill:none;stroke:#0076BA;stroke-width:6;" d="
|
||||
M781,353.4h358c8.4,0,13.4,0,16.7,1.4c4.8,1.8,8.6,5.6,10.4,10.4c1.4,3.3,1.4,8.4,1.4,16.7v268.8c0,8.4,0,13.4-1.4,16.7
|
||||
c-1.8,4.8-5.6,8.6-10.4,10.4c-3.3,1.4-8.4,1.4-16.7,1.4H781c-8.4,0-13.4,0-16.7-1.4c-4.8-1.8-8.6-5.6-10.4-10.4
|
||||
c-1.4-3.3-1.4-8.4-1.4-16.7V381.9c0-8.4,0-13.4,1.4-16.7c1.8-4.8,5.6-8.6,10.4-10.4C767.6,353.4,772.6,353.4,781,353.4z"/>
|
||||
</g>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_00000039115730048533150280000004801426134829330596_" x="737.4" y="353.4" width="445.2" height="325.8"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000016774010820287457580000002865762276517841055_">
|
||||
<use xlink:href="#SVGID_00000039115730048533150280000004801426134829330596_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
<g style="clip-path:url(#SVGID_00000016774010820287457580000002865762276517841055_);">
|
||||
<text transform="matrix(1 0 0 1 877.1631 540.0012)" class="st4 st5 st6">ESP32</text>
|
||||
</g>
|
||||
</g>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_00000075151113810300876960000013536007874996673433_" width="1920" height="1080"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000107586269070498659130000000925674275700912771_">
|
||||
<use xlink:href="#SVGID_00000075151113810300876960000013536007874996673433_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000107586269070498659130000000925674275700912771_);fill:none;stroke:#0076BA;stroke-width:6;" d="
|
||||
M107.1,424.9h369.6c6.7,0,10.7,0,13.4,1.1c3.9,1.4,6.9,4.4,8.3,8.3c1.1,2.7,1.1,6.7,1.1,13.4v153.2c0,6.7,0,10.7-1.1,13.4
|
||||
c-1.4,3.9-4.4,6.9-8.3,8.3c-2.7,1.1-6.7,1.1-13.4,1.1H107.1c-6.7,0-10.7,0-13.4-1.1c-3.9-1.4-6.9-4.4-8.3-8.3
|
||||
c-1.1-2.7-1.1-6.7-1.1-13.4V447.6c0-6.7,0-10.7,1.1-13.4c1.4-3.9,4.4-6.9,8.3-8.3C96.4,424.9,100.4,424.9,107.1,424.9z"/>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000107586269070498659130000000925674275700912771_);fill:none;stroke:#0076BA;stroke-width:6;stroke-dasharray:12,12;" d="
|
||||
M777.9,846.9h364.3c7.5,0,11.9,0,14.9,1.2c4.3,1.6,7.7,5,9.3,9.3c1.2,3,1.2,7.5,1.2,14.9V974c0,7.5,0,11.9-1.2,14.9
|
||||
c-1.6,4.3-5,7.7-9.3,9.3c-3,1.2-7.5,1.2-14.9,1.2H777.9c-7.5,0-11.9,0-14.9-1.2c-4.3-1.6-7.7-5-9.3-9.3c-1.2-3-1.2-7.5-1.2-14.9
|
||||
V872.4c0-7.5,0-11.9,1.2-14.9c1.6-4.3,5-7.7,9.3-9.3C765.9,846.9,770.4,846.9,777.9,846.9z"/>
|
||||
</g>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_00000128454988870213717900000008065144461254055839_" x="737.3" y="846.9" width="445.4" height="152.5"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000124119398570885020400000016506660910002275730_">
|
||||
<use xlink:href="#SVGID_00000128454988870213717900000008065144461254055839_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
<g style="clip-path:url(#SVGID_00000124119398570885020400000016506660910002275730_);">
|
||||
<text transform="matrix(1 0 0 1 802.3304 947.0799)"><tspan x="0" y="0" class="st5 st6">RC </tspan><tspan x="92" y="0" class="st5 st6 st10">R</tspan><tspan x="128.2" y="0" class="st5 st6">ecei</tspan><tspan x="232.7" y="0" class="st5 st6">v</tspan><tspan x="262.2" y="0" class="st5 st6">er</tspan></text>
|
||||
</g>
|
||||
</g>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_00000130607252134697782620000007941153577788369085_" width="1920" height="1080"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000168829390248542060830000011029723519784449157_">
|
||||
<use xlink:href="#SVGID_00000130607252134697782620000007941153577788369085_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);fill:none;stroke:#D5D5D5;stroke-width:6;" d="
|
||||
M429.1,526.4c96.1,19.6,194.9,26.9,296.4,22l3-0.2"/>
|
||||
<polygon style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);fill:#D5D5D5;" points="726.2,560.4
|
||||
749.5,547.1 724.8,536.5 "/>
|
||||
<g style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);">
|
||||
<text transform="matrix(1 0 0 1 571.0244 599.9149)" class="st5 st14">SPI</text>
|
||||
</g>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);fill:none;stroke:#D5D5D5;stroke-width:6;stroke-dasharray:12,12;" d="
|
||||
M991.9,843.9c7.9-38.5,9.3-84.5,4.4-137.8l-0.3-3"/>
|
||||
<polygon style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);fill:#D5D5D5;" points="1008.2,704.8
|
||||
993.7,682.3 984.4,707.4 "/>
|
||||
<g style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);">
|
||||
<text transform="matrix(1 0 0 1 746.5599 778.8257)"><tspan x="0" y="0" class="st5 st14">SBUS (</tspan><tspan x="122.2" y="0" class="st5 st14">U</tspan><tspan x="148.3" y="0" class="st5 st14">A</tspan><tspan x="172.2" y="0" class="st5 st14 st10">R</tspan><tspan x="196" y="0" class="st5 st14">T</tspan><tspan x="219.8" y="0" class="st5 st14">)</tspan></text>
|
||||
</g>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);fill:none;stroke:#D5D5D5;stroke-width:6;" d="
|
||||
M1170.5,537.8c72.8,3.7,147.2,3.8,223,0.3l3-0.2"/>
|
||||
<polygon style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);fill:#D5D5D5;" points="1394.1,550
|
||||
1417.5,536.8 1392.9,526 "/>
|
||||
<g style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);">
|
||||
<text transform="matrix(1 0 0 1 1236.025 523.2462)" class="st5 st14">PWM</text>
|
||||
</g>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);fill:none;stroke:#D5D5D5;stroke-width:6;" d="
|
||||
M1612.3,595.6c-10.7,73.2-11.3,149.1-2,227.5l0.4,3"/>
|
||||
<polygon style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);fill:#D5D5D5;" points="1598.5,824.7
|
||||
1613.5,846.9 1622.2,821.6 "/>
|
||||
<g style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);">
|
||||
<text transform="matrix(1 0 0 1 1450.277 736.9998)"><tspan x="0" y="0" class="st5 st14 st10">V</tspan><tspan x="22.1" y="0" class="st5 st14">o</tspan><tspan x="43.8" y="0" class="st5 st14">l</tspan><tspan x="53" y="0" class="st5 st14">tage</tspan></text>
|
||||
</g>
|
||||
<g style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);">
|
||||
<text transform="matrix(1 0 0 1 212.8847 595.784)"><tspan x="0" y="0" class="st5 st6">G</tspan><tspan x="40" y="0" class="st5 st6 st16">Y</tspan><tspan x="70.8" y="0" class="st5 st6">-91</tspan></text>
|
||||
</g>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000168829390248542060830000011029723519784449157_);fill:none;stroke:#0076BA;stroke-width:6;" d="
|
||||
M777.9,77.5h364.3c7.5,0,11.9,0,14.9,1.2c4.3,1.6,7.7,5,9.3,9.3c1.2,3,1.2,7.5,1.2,14.9v101.7c0,7.5,0,11.9-1.2,14.9
|
||||
c-1.6,4.3-5,7.7-9.3,9.3c-3,1.2-7.5,1.2-14.9,1.2H777.9c-7.5,0-11.9,0-14.9-1.2c-4.3-1.6-7.7-5-9.3-9.3c-1.2-3-1.2-7.5-1.2-14.9
|
||||
V103c0-7.5,0-11.9,1.2-14.9c1.6-4.3,5-7.7,9.3-9.3C765.9,77.5,770.4,77.5,777.9,77.5z"/>
|
||||
</g>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_00000137820956330754408580000017691839315522467728_" x="737.3" y="77.5" width="445.4" height="152.5"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000016789657776342930330000013964875638603798149_">
|
||||
<use xlink:href="#SVGID_00000137820956330754408580000017691839315522467728_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
<g style="clip-path:url(#SVGID_00000016789657776342930330000013964875638603798149_);">
|
||||
<text transform="matrix(1 0 0 1 865.0551 177.6901)"><tspan x="0" y="0" class="st5 st6">B</tspan><tspan x="35.6" y="0" class="st5 st6">a</tspan><tspan x="67.1" y="0" class="st5 st6">t</tspan><tspan x="86.7" y="0" class="st5 st6">tery</tspan></text>
|
||||
</g>
|
||||
</g>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_00000016756559387490335270000000552908433054081441_" width="1920" height="1080"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000171720285722730791730000010936644841389779363_">
|
||||
<use xlink:href="#SVGID_00000016756559387490335270000000552908433054081441_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000171720285722730791730000010936644841389779363_);fill:none;stroke:#D5D5D5;stroke-width:6;" d="
|
||||
M925.7,233.1c-5.3,27.2-6.8,58.4-4.3,93.4l0.3,3"/>
|
||||
<polygon style="clip-path:url(#SVGID_00000171720285722730791730000010936644841389779363_);fill:#D5D5D5;" points="909.4,327.6
|
||||
923.5,350.4 933.3,325.4 "/>
|
||||
<g style="clip-path:url(#SVGID_00000171720285722730791730000010936644841389779363_);">
|
||||
<text transform="matrix(1 0 0 1 937.7745 299.4789)"><tspan x="0" y="0" class="st5 st14">>3</tspan><tspan x="50.9" y="0" class="st5 st14">.</tspan><tspan x="62.1" y="0" class="st5 st14">7V</tspan></text>
|
||||
</g>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000171720285722730791730000010936644841389779363_);fill:none;stroke:#D5D5D5;stroke-width:6;" d="
|
||||
M1170.6,161.5c228,25.6,371.6,110,430.8,253.1l1.1,2.8"/>
|
||||
<polygon style="clip-path:url(#SVGID_00000171720285722730791730000010936644841389779363_);fill:#D5D5D5;" points="1590.2,418.8
|
||||
1609.8,437.1 1612.6,410.4 "/>
|
||||
<g style="clip-path:url(#SVGID_00000171720285722730791730000010936644841389779363_);">
|
||||
<text transform="matrix(1 0 0 1 1396.229 204.8126)"><tspan x="0" y="0" class="st5 st14">>3</tspan><tspan x="50.9" y="0" class="st5 st14">.</tspan><tspan x="62.1" y="0" class="st5 st14">7V</tspan></text>
|
||||
</g>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000171720285722730791730000010936644841389779363_);fill:none;stroke:#FF9300;stroke-width:6;" d="
|
||||
M1445.9,849.9h364.3c7.5,0,11.9,0,14.9,1.2c4.3,1.6,7.7,5,9.3,9.3c1.2,3,1.2,7.5,1.2,14.9V977c0,7.5,0,11.9-1.2,14.9
|
||||
c-1.6,4.3-5,7.7-9.3,9.3c-3,1.2-7.5,1.2-14.9,1.2h-364.3c-7.5,0-11.9,0-14.9-1.2c-4.3-1.6-7.7-5-9.3-9.3c-1.2-3-1.2-7.5-1.2-14.9
|
||||
V875.4c0-7.5,0-11.9,1.2-14.9c1.6-4.3,5-7.7,9.3-9.3C1434,849.9,1438.5,849.9,1445.9,849.9z"/>
|
||||
</g>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_00000125590519588794066770000004908208256101094292_" x="1405.4" y="849.9" width="445.4" height="152.5"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000020367246295901477730000005237269270844408247_">
|
||||
<use xlink:href="#SVGID_00000125590519588794066770000004908208256101094292_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
<g style="clip-path:url(#SVGID_00000020367246295901477730000005237269270844408247_);">
|
||||
<text transform="matrix(1 0 0 1 1507.9561 950.0799)" class="st5 st6">Motors </text>
|
||||
</g>
|
||||
<g style="clip-path:url(#SVGID_00000020367246295901477730000005237269270844408247_);">
|
||||
<text transform="matrix(1 0 0 1 1706.5596 950.0799)" class="st5 st14">x4</text>
|
||||
</g>
|
||||
</g>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_00000078757851010072822240000012642622956562684587_" width="1920" height="1080"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000132802581807603645050000011682039405646066589_">
|
||||
<use xlink:href="#SVGID_00000078757851010072822240000012642622956562684587_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000132802581807603645050000011682039405646066589_);fill:none;stroke:#0076BA;stroke-width:6;" d="
|
||||
M1445.9,440.1h364.3c7.5,0,11.9,0,14.9,1.2c4.3,1.6,7.7,5,9.3,9.3c1.2,3,1.2,7.5,1.2,14.9v101.7c0,7.5,0,11.9-1.2,14.9
|
||||
c-1.6,4.3-5,7.7-9.3,9.3c-3,1.2-7.5,1.2-14.9,1.2h-364.3c-7.5,0-11.9,0-14.9-1.2c-4.3-1.6-7.7-5-9.3-9.3c-1.2-3-1.2-7.5-1.2-14.9
|
||||
V465.5c0-7.5,0-11.9,1.2-14.9c1.6-4.3,5-7.7,9.3-9.3C1434,440.1,1438.5,440.1,1445.9,440.1z"/>
|
||||
</g>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_00000121968229548182703620000008187702022402881724_" x="1405.4" y="440.1" width="445.4" height="152.5"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000130637550574198708210000000321319364498620830_">
|
||||
<use xlink:href="#SVGID_00000121968229548182703620000008187702022402881724_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
<g style="clip-path:url(#SVGID_00000130637550574198708210000000321319364498620830_);">
|
||||
<text transform="matrix(1 0 0 1 1486.833 540.213)" class="st5 st6">MOSFE</text>
|
||||
</g>
|
||||
<g style="clip-path:url(#SVGID_00000130637550574198708210000000321319364498620830_);">
|
||||
<text transform="matrix(1 0 0 1 1673.8936 540.213)" class="st26 st5 st6">T</text>
|
||||
</g>
|
||||
<g style="clip-path:url(#SVGID_00000130637550574198708210000000321319364498620830_);">
|
||||
<text transform="matrix(1 0 0 1 1708.9326 540.213)" class="st5 st6"> </text>
|
||||
</g>
|
||||
<g style="clip-path:url(#SVGID_00000130637550574198708210000000321319364498620830_);">
|
||||
<text transform="matrix(1 0 0 1 1727.6826 540.213)" class="st5 st14">x4</text>
|
||||
</g>
|
||||
</g>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_00000038395453607794357290000016650519059911045024_" width="1920" height="1080"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000077303055156502403190000000114138446408493755_">
|
||||
<use xlink:href="#SVGID_00000038395453607794357290000016650519059911045024_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
|
||||
<path style="clip-path:url(#SVGID_00000077303055156502403190000000114138446408493755_);fill:none;stroke:#0076BA;stroke-width:3;" d="
|
||||
M167.8,451H416c4.5,0,7.1,0,8.9,0.7c2.6,0.9,4.6,3,5.5,5.5c0.7,1.8,0.7,4.5,0.7,8.9v46.6c0,4.5,0,7.1-0.7,8.9
|
||||
c-0.9,2.6-3,4.6-5.5,5.5c-1.8,0.7-4.5,0.7-8.9,0.7H167.8c-4.5,0-7.1,0-8.9-0.7c-2.6-0.9-4.6-3-5.5-5.5c-0.7-1.8-0.7-4.5-0.7-8.9
|
||||
v-46.6c0-4.5,0-7.1,0.7-8.9c0.9-2.6,3-4.6,5.5-5.5C160.7,451,163.4,451,167.8,451z"/>
|
||||
</g>
|
||||
<g>
|
||||
<defs>
|
||||
<rect id="SVGID_00000077301502947632695740000005629016913768095395_" x="145.2" y="451" width="293.5" height="77"/>
|
||||
</defs>
|
||||
<clipPath id="SVGID_00000004518590643829462700000008680026774255780777_">
|
||||
<use xlink:href="#SVGID_00000077301502947632695740000005629016913768095395_" style="overflow:visible;"/>
|
||||
</clipPath>
|
||||
<g style="clip-path:url(#SVGID_00000004518590643829462700000008680026774255780777_);">
|
||||
<text transform="matrix(1 0 0 1 197.8359 501.2101)" class="st5 st29">MPU9250 IMU</text>
|
||||
</g>
|
||||
</g>
|
||||
</svg>
|
||||
|
After Width: | Height: | Size: 17 KiB |
BIN
docs/img/screw-m1.4.jpg
Normal file
|
After Width: | Height: | Size: 30 KiB |
BIN
docs/img/screw-m3.jpg
Normal file
|
After Width: | Height: | Size: 3.7 KiB |
BIN
docs/img/washer-m3.jpg
Normal file
|
After Width: | Height: | Size: 5.9 KiB |
BIN
docs/img/wire-28awg.jpg
Normal file
|
After Width: | Height: | Size: 37 KiB |
72
docs/log.md
Normal file
@@ -0,0 +1,72 @@
|
||||
# Log analysis
|
||||
|
||||
Flix quadcopter uses RAM to store flight log data. The default log capacity is 10 seconds at 100 Hz. This configuration can be adjusted in the `log.ino` file.
|
||||
|
||||
To perform log analysis, you need to download the log right after the flight without powering off the drone. Then you can use several tools to analyze the log data.
|
||||
|
||||
## Log download
|
||||
|
||||
To download the log, connect the ESP32 using USB right after the flight and run the following command:
|
||||
|
||||
```bash
|
||||
make log
|
||||
```
|
||||
|
||||
Logs are stored in `tools/log/*.csv` files.
|
||||
|
||||
## Analysis
|
||||
|
||||
### PlotJuggler
|
||||
|
||||
The recommended tool for log analysis is PlotJuggler.
|
||||
|
||||
<img src="img/plotjuggler.png" width="500">
|
||||
|
||||
1. Install PlotJuggler using the [official instructions](https://github.com/facontidavide/PlotJuggler?tab=readme-ov-file#installation).
|
||||
|
||||
2. Run PlotJuggler and drag'n'drop the downloaded log file there. Choose `t` column to be used as X axis.
|
||||
|
||||
You can open the most recent downloaded file using the command:
|
||||
|
||||
```bash
|
||||
make plot
|
||||
```
|
||||
|
||||
You can perform both log download and run PlotJuggler in one command:
|
||||
|
||||
```bash
|
||||
make log plot
|
||||
```
|
||||
|
||||
### FlightPlot
|
||||
|
||||
FlightPlot is a powerful tool for analyzing logs in [ULog format](https://docs.px4.io/main/en/dev_log/ulog_file_format.html). This format is used in PX4 and ArduPilot flight software.
|
||||
|
||||
<img src="img/flightplot.png" width="500">
|
||||
|
||||
1. [Install FlightPlot](https://github.com/PX4/FlightPlot).
|
||||
2. Flix repository contains a tool for converting CSV logs to ULog format. Build the tool using [the instructions](../tools/csv_to_ulog/README.md) and convert the log you want to analyze.
|
||||
3. Run FlightPlot and drag'n'drop the converted ULog-file there.
|
||||
|
||||
### Foxglove Studio
|
||||
|
||||
Foxglove is a tool for visualizing and analyzing robotics data with very rich functionality. It can import various formats, but mainly focuses on its own format, called [MCAP](https://mcap.dev).
|
||||
|
||||
<img src="img/foxglove.png" width="500">
|
||||
|
||||
1. Install Foxglove Studio from the [official website](https://foxglove.dev/download).
|
||||
|
||||
2. Flix repository contains a tool for converting CSV logs to MCAP format. First, install its dependencies:
|
||||
|
||||
```bash
|
||||
cd tools
|
||||
pip install -r requirements.txt
|
||||
```
|
||||
|
||||
3. Convert the log you want to analyze:
|
||||
|
||||
```bash
|
||||
csv_to_mcap.py log_file.csv
|
||||
```
|
||||
|
||||
4. Open the log in Foxglove Studio using *Open local file* command.
|
||||
30
docs/version0.md
Normal file
@@ -0,0 +1,30 @@
|
||||
# Flix version 0
|
||||
|
||||
Flix version 0 (obsolete):
|
||||
|
||||
<img src="img/flix.jpg" width=500 alt="Flix quadcopter">
|
||||
|
||||
## Components list
|
||||
|
||||
|Type|Part|Image|Quantity|
|
||||
|-|-|-|-|
|
||||
|Microcontroller board|ESP32 Mini|<img src="img/esp32.jpg" width=100>|1|
|
||||
|IMU and barometer² board|GY-91 (or other MPU-9250 board)|<img src="img/gy-91.jpg" width=100>|1|
|
||||
|Quadcopter frame|K100|<img src="img/frame.jpg" width=100>|1|
|
||||
|Motor|8520 3.7V brushed motor (**shaft 0.8mm!**)|<img src="img/motor.jpeg" width=100>|4|
|
||||
|Propeller|Hubsan 55 mm|<img src="img/prop.jpg" width=100>|4|
|
||||
|Motor ESC|2.7A 1S Dual Way Micro Brush ESC|<img src="img/esc.jpg" width=100>|4|
|
||||
|RC transmitter|KINGKONG TINY X8|<img src="img/tx.jpg" width=100>|1|
|
||||
|RC receiver|DF500 (SBUS)|<img src="img/rx.jpg" width=100>|1|
|
||||
|~~SBUS inverter~~*||<img src="img/inv.jpg" width=100>|~~1~~|
|
||||
|Battery|3.7 Li-Po 850 MaH 60C|||
|
||||
|Battery charger||<img src="img/charger.jpg" width=100>|1|
|
||||
|Wires, connectors, tape, ...|||
|
||||
|
||||
*\* — not needed as ESP32 supports [software pin inversion](https://github.com/bolderflight/sbus#inverted-serial).*
|
||||
|
||||
## Schematics
|
||||
|
||||
<img src="img/schematics.svg" width=800 alt="Flix schematics">
|
||||
|
||||
You can also check a user contributed [variant of complete circuit diagram](https://miro.com/app/board/uXjVN-dTjoo=/) of the drone.
|
||||
29
flix/cli.ino
@@ -27,9 +27,11 @@ const char* motd =
|
||||
"rc - show RC data\n"
|
||||
"mot - show motor data\n"
|
||||
"log - dump in-RAM log\n"
|
||||
"cr - calibrate RC\n"
|
||||
"cg - calibrate gyro\n"
|
||||
"ca - calibrate accel\n"
|
||||
"fullmot <n> - test motor on all signals\n"
|
||||
"mfr, mfl, mrr, mrl - test appropriate motor\n"
|
||||
"fullmot <n> - full motor test\n"
|
||||
"reset - reset drone's state\n";
|
||||
|
||||
const struct Param {
|
||||
@@ -57,8 +59,7 @@ const struct Param {
|
||||
{"t", &t, nullptr},
|
||||
};
|
||||
|
||||
void doCommand(String& command, String& value)
|
||||
{
|
||||
void doCommand(String& command, String& value) {
|
||||
if (command == "help" || command == "motd") {
|
||||
Serial.println(motd);
|
||||
} else if (command == "show") {
|
||||
@@ -72,19 +73,22 @@ void doCommand(String& command, String& value)
|
||||
Serial.printf("gyro: %f %f %f\n", rates.x, rates.y, rates.z);
|
||||
Serial.printf("acc: %f %f %f\n", acc.x, acc.y, acc.z);
|
||||
printIMUCal();
|
||||
Serial.printf("frequency: %f\n", loopFreq);
|
||||
} else if (command == "rc") {
|
||||
Serial.printf("Raw: throttle %d yaw %d pitch %d roll %d aux %d mode %d\n",
|
||||
Serial.printf("Raw: throttle %d yaw %d pitch %d roll %d armed %d mode %d\n",
|
||||
channels[RC_CHANNEL_THROTTLE], channels[RC_CHANNEL_YAW], channels[RC_CHANNEL_PITCH],
|
||||
channels[RC_CHANNEL_ROLL], channels[RC_CHANNEL_AUX], channels[RC_CHANNEL_MODE]);
|
||||
Serial.printf("Control: throttle %f yaw %f pitch %f roll %f aux %f mode %f\n",
|
||||
channels[RC_CHANNEL_ROLL], channels[RC_CHANNEL_ARMED], channels[RC_CHANNEL_MODE]);
|
||||
Serial.printf("Control: throttle %f yaw %f pitch %f roll %f armed %f mode %f\n",
|
||||
controls[RC_CHANNEL_THROTTLE], controls[RC_CHANNEL_YAW], controls[RC_CHANNEL_PITCH],
|
||||
controls[RC_CHANNEL_ROLL], controls[RC_CHANNEL_AUX], controls[RC_CHANNEL_MODE]);
|
||||
controls[RC_CHANNEL_ROLL], controls[RC_CHANNEL_ARMED], controls[RC_CHANNEL_MODE]);
|
||||
Serial.printf("Mode: %s\n", getModeName());
|
||||
} else if (command == "mot") {
|
||||
Serial.printf("MOTOR front-right %f front-left %f rear-right %f rear-left %f\n",
|
||||
motors[MOTOR_FRONT_RIGHT], motors[MOTOR_FRONT_LEFT], motors[MOTOR_REAR_RIGHT], motors[MOTOR_REAR_LEFT]);
|
||||
} else if (command == "log") {
|
||||
dumpLog();
|
||||
} else if (command == "cr") {
|
||||
calibrateRC();
|
||||
} else if (command == "cg") {
|
||||
calibrateGyro();
|
||||
} else if (command == "ca") {
|
||||
@@ -98,7 +102,7 @@ void doCommand(String& command, String& value)
|
||||
} else if (command == "mrl") {
|
||||
cliTestMotor(MOTOR_REAR_LEFT);
|
||||
} else if (command == "fullmot") {
|
||||
fullMotorTest(value.toInt(), false);
|
||||
fullMotorTest(value.toInt());
|
||||
} else if (command == "reset") {
|
||||
attitude = Quaternion();
|
||||
} else {
|
||||
@@ -119,8 +123,7 @@ void doCommand(String& command, String& value)
|
||||
}
|
||||
}
|
||||
|
||||
void showTable()
|
||||
{
|
||||
void showTable() {
|
||||
for (uint8_t i = 0; i < sizeof(params) / sizeof(params[0]); i++) {
|
||||
Serial.print(params[i].name);
|
||||
Serial.print(" ");
|
||||
@@ -128,8 +131,7 @@ void showTable()
|
||||
}
|
||||
}
|
||||
|
||||
void cliTestMotor(uint8_t n)
|
||||
{
|
||||
void cliTestMotor(uint8_t n) {
|
||||
Serial.printf("Testing motor %d\n", n);
|
||||
motors[n] = 1;
|
||||
sendMotors();
|
||||
@@ -139,8 +141,7 @@ void cliTestMotor(uint8_t n)
|
||||
Serial.println("Done");
|
||||
}
|
||||
|
||||
void parseInput()
|
||||
{
|
||||
void parseInput() {
|
||||
static bool showMotd = true;
|
||||
static String command;
|
||||
static String value;
|
||||
|
||||
@@ -27,15 +27,14 @@
|
||||
#define PITCH_I ROLL_I
|
||||
#define PITCH_D ROLL_D
|
||||
#define YAW_P 3
|
||||
#define PITCHRATE_MAX 360 * DEG_TO_RAD
|
||||
#define ROLLRATE_MAX 360 * DEG_TO_RAD
|
||||
#define YAWRATE_MAX 360 * DEG_TO_RAD
|
||||
#define MAX_TILT 30 * DEG_TO_RAD
|
||||
#define PITCHRATE_MAX radians(360)
|
||||
#define ROLLRATE_MAX radians(360)
|
||||
#define YAWRATE_MAX radians(360)
|
||||
#define MAX_TILT radians(30)
|
||||
|
||||
#define RATES_LFP_ALPHA 0.8 // cutoff frequency ~ 250 Hz
|
||||
#define RATES_D_LPF_ALPHA 0.2 // cutoff frequency ~ 40 Hz
|
||||
|
||||
enum { MANUAL, ACRO, STAB } mode = STAB;
|
||||
enum { MANUAL, ACRO, STAB, USER } mode = STAB;
|
||||
enum { YAW, YAW_RATE } yawMode = YAW;
|
||||
bool armed = false;
|
||||
|
||||
@@ -46,15 +45,12 @@ PID rollPID(ROLL_P, ROLL_I, ROLL_D);
|
||||
PID pitchPID(PITCH_P, PITCH_I, PITCH_D);
|
||||
PID yawPID(YAW_P, 0, 0);
|
||||
|
||||
LowPassFilter<Vector> ratesFilter(RATES_LFP_ALPHA);
|
||||
|
||||
Quaternion attitudeTarget;
|
||||
Vector ratesTarget;
|
||||
Vector torqueTarget;
|
||||
float thrustTarget;
|
||||
|
||||
void control()
|
||||
{
|
||||
void control() {
|
||||
interpretRC();
|
||||
if (mode == STAB) {
|
||||
controlAttitude();
|
||||
@@ -68,38 +64,39 @@ void control()
|
||||
}
|
||||
}
|
||||
|
||||
void interpretRC()
|
||||
{
|
||||
void interpretRC() {
|
||||
armed = controls[RC_CHANNEL_THROTTLE] >= 0.05 && controls[RC_CHANNEL_ARMED] >= 0.5;
|
||||
|
||||
// NOTE: put ACRO or MANUAL modes there if you want to use them
|
||||
if (controls[RC_CHANNEL_MODE] < 0.25) {
|
||||
mode = MANUAL;
|
||||
mode = STAB;
|
||||
} else if (controls[RC_CHANNEL_MODE] < 0.75) {
|
||||
mode = ACRO;
|
||||
mode = STAB;
|
||||
} else {
|
||||
mode = STAB;
|
||||
}
|
||||
|
||||
armed = controls[RC_CHANNEL_THROTTLE] >= 0.05 && controls[RC_CHANNEL_AUX] >= 0.5;
|
||||
thrustTarget = controls[RC_CHANNEL_THROTTLE];
|
||||
|
||||
if (mode == ACRO) {
|
||||
yawMode = YAW_RATE;
|
||||
ratesTarget.x = controls[RC_CHANNEL_ROLL] * ROLLRATE_MAX;
|
||||
ratesTarget.y = -controls[RC_CHANNEL_PITCH] * PITCHRATE_MAX; // up pitch stick means tilt clockwise in frd
|
||||
ratesTarget.z = controls[RC_CHANNEL_YAW] * YAWRATE_MAX;
|
||||
ratesTarget.y = controls[RC_CHANNEL_PITCH] * PITCHRATE_MAX;
|
||||
ratesTarget.z = -controls[RC_CHANNEL_YAW] * YAWRATE_MAX; // positive yaw stick means clockwise rotation in FLU
|
||||
|
||||
} else if (mode == STAB) {
|
||||
yawMode = controls[RC_CHANNEL_YAW] == 0 ? YAW : YAW_RATE;
|
||||
|
||||
attitudeTarget = Quaternion::fromEulerZYX(
|
||||
attitudeTarget = Quaternion::fromEulerZYX(Vector(
|
||||
controls[RC_CHANNEL_ROLL] * MAX_TILT,
|
||||
-controls[RC_CHANNEL_PITCH] * MAX_TILT,
|
||||
attitudeTarget.getYaw());
|
||||
ratesTarget.z = controls[RC_CHANNEL_YAW] * YAWRATE_MAX;
|
||||
controls[RC_CHANNEL_PITCH] * MAX_TILT,
|
||||
attitudeTarget.getYaw()));
|
||||
ratesTarget.z = -controls[RC_CHANNEL_YAW] * YAWRATE_MAX; // positive yaw stick means clockwise rotation in FLU
|
||||
|
||||
} else if (mode == MANUAL) {
|
||||
// passthrough mode
|
||||
yawMode = YAW_RATE;
|
||||
torqueTarget = Vector(controls[RC_CHANNEL_ROLL], -controls[RC_CHANNEL_PITCH], controls[RC_CHANNEL_YAW]) * 0.01;
|
||||
torqueTarget = Vector(controls[RC_CHANNEL_ROLL], controls[RC_CHANNEL_PITCH], -controls[RC_CHANNEL_YAW]) * 0.01;
|
||||
}
|
||||
|
||||
if (yawMode == YAW_RATE || !motorsActive()) {
|
||||
@@ -108,8 +105,7 @@ void interpretRC()
|
||||
}
|
||||
}
|
||||
|
||||
void controlAttitude()
|
||||
{
|
||||
void controlAttitude() {
|
||||
if (!armed) {
|
||||
rollPID.reset();
|
||||
pitchPID.reset();
|
||||
@@ -117,7 +113,7 @@ void controlAttitude()
|
||||
return;
|
||||
}
|
||||
|
||||
const Vector up(0, 0, -1);
|
||||
const Vector up(0, 0, 1);
|
||||
Vector upActual = attitude.rotate(up);
|
||||
Vector upTarget = attitudeTarget.rotate(up);
|
||||
|
||||
@@ -127,12 +123,12 @@ void controlAttitude()
|
||||
ratesTarget.y = pitchPID.update(error.y, dt);
|
||||
|
||||
if (yawMode == YAW) {
|
||||
ratesTarget.z = yawPID.update(wrapAngle(attitudeTarget.getYaw() - attitude.getYaw()), dt);
|
||||
float yawError = wrapAngle(attitudeTarget.getYaw() - attitude.getYaw());
|
||||
ratesTarget.z = yawPID.update(yawError, dt);
|
||||
}
|
||||
}
|
||||
|
||||
void controlRate()
|
||||
{
|
||||
void controlRate() {
|
||||
if (!armed) {
|
||||
rollRatePID.reset();
|
||||
pitchRatePID.reset();
|
||||
@@ -140,24 +136,24 @@ void controlRate()
|
||||
return;
|
||||
}
|
||||
|
||||
Vector ratesFiltered = ratesFilter.update(rates);
|
||||
Vector error = ratesTarget - rates;
|
||||
|
||||
torqueTarget.x = rollRatePID.update(ratesTarget.x - ratesFiltered.x, dt); // un-normalized "torque"
|
||||
torqueTarget.y = pitchRatePID.update(ratesTarget.y - ratesFiltered.y, dt);
|
||||
torqueTarget.z = yawRatePID.update(ratesTarget.z - ratesFiltered.z, dt);
|
||||
// Calculate desired torque, where 0 - no torque, 1 - maximum possible torque
|
||||
torqueTarget.x = rollRatePID.update(error.x, dt);
|
||||
torqueTarget.y = pitchRatePID.update(error.y, dt);
|
||||
torqueTarget.z = yawRatePID.update(error.z, dt);
|
||||
}
|
||||
|
||||
void controlTorque()
|
||||
{
|
||||
void controlTorque() {
|
||||
if (!armed) {
|
||||
memset(motors, 0, sizeof(motors));
|
||||
return;
|
||||
}
|
||||
|
||||
motors[MOTOR_FRONT_LEFT] = thrustTarget + torqueTarget.y + torqueTarget.x - torqueTarget.z;
|
||||
motors[MOTOR_FRONT_RIGHT] = thrustTarget + torqueTarget.y - torqueTarget.x + torqueTarget.z;
|
||||
motors[MOTOR_REAR_LEFT] = thrustTarget - torqueTarget.y + torqueTarget.x + torqueTarget.z;
|
||||
motors[MOTOR_REAR_RIGHT] = thrustTarget - torqueTarget.y - torqueTarget.x - torqueTarget.z;
|
||||
motors[MOTOR_FRONT_LEFT] = thrustTarget + torqueTarget.x - torqueTarget.y + torqueTarget.z;
|
||||
motors[MOTOR_FRONT_RIGHT] = thrustTarget - torqueTarget.x - torqueTarget.y - torqueTarget.z;
|
||||
motors[MOTOR_REAR_LEFT] = thrustTarget + torqueTarget.x + torqueTarget.y - torqueTarget.z;
|
||||
motors[MOTOR_REAR_RIGHT] = thrustTarget - torqueTarget.x + torqueTarget.y + torqueTarget.z;
|
||||
|
||||
motors[0] = constrain(motors[0], 0, 1);
|
||||
motors[1] = constrain(motors[1], 0, 1);
|
||||
@@ -165,17 +161,16 @@ void controlTorque()
|
||||
motors[3] = constrain(motors[3], 0, 1);
|
||||
}
|
||||
|
||||
bool motorsActive()
|
||||
{
|
||||
bool motorsActive() {
|
||||
return motors[0] > 0 || motors[1] > 0 || motors[2] > 0 || motors[3] > 0;
|
||||
}
|
||||
|
||||
const char* getModeName()
|
||||
{
|
||||
const char* getModeName() {
|
||||
switch (mode) {
|
||||
case MANUAL: return "MANUAL";
|
||||
case ACRO: return "ACRO";
|
||||
case STAB: return "STAB";
|
||||
case USER: return "USER";
|
||||
default: return "UNKNOWN";
|
||||
}
|
||||
}
|
||||
|
||||
@@ -5,28 +5,30 @@
|
||||
|
||||
#include "quaternion.h"
|
||||
#include "vector.h"
|
||||
#include "lpf.h"
|
||||
|
||||
#define ONE_G 9.807f
|
||||
#define ACC_MIN 0.9f
|
||||
#define ACC_MAX 1.1f
|
||||
#define WEIGHT_ACC 0.5f
|
||||
#define RATES_LFP_ALPHA 0.2 // cutoff frequency ~ 40 Hz
|
||||
|
||||
void estimate()
|
||||
{
|
||||
LowPassFilter<Vector> ratesFilter(RATES_LFP_ALPHA);
|
||||
|
||||
void estimate() {
|
||||
applyGyro();
|
||||
applyAcc();
|
||||
signalizeHorizontality();
|
||||
}
|
||||
|
||||
void applyGyro()
|
||||
{
|
||||
// applying gyro
|
||||
void applyGyro() {
|
||||
// filter gyro to get angular rates
|
||||
rates = ratesFilter.update(gyro);
|
||||
|
||||
// apply rates to attitude
|
||||
attitude *= Quaternion::fromAngularRates(rates * dt);
|
||||
attitude.normalize();
|
||||
}
|
||||
|
||||
void applyAcc()
|
||||
{
|
||||
void applyAcc() {
|
||||
// test should we apply accelerometer gravity correction
|
||||
float accNorm = acc.norm();
|
||||
bool landed = !motorsActive() && abs(accNorm - ONE_G) < ONE_G * 0.1f;
|
||||
@@ -34,7 +36,7 @@ void applyAcc()
|
||||
if (!landed) return;
|
||||
|
||||
// calculate accelerometer correction
|
||||
Vector up = attitude.rotate(Vector(0, 0, -1));
|
||||
Vector up = attitude.rotate(Vector(0, 0, 1));
|
||||
Vector correction = Vector::angularRatesBetweenVectors(acc, up) * dt * WEIGHT_ACC;
|
||||
|
||||
// apply correction
|
||||
@@ -42,8 +44,7 @@ void applyAcc()
|
||||
attitude.normalize();
|
||||
}
|
||||
|
||||
void signalizeHorizontality()
|
||||
{
|
||||
float angle = Vector::angleBetweenVectors(attitude.rotate(Vector(0, 0, -1)), Vector(0, 0, -1));
|
||||
setLED(angle < 15 * DEG_TO_RAD);
|
||||
void signalizeHorizontality() {
|
||||
float angle = Vector::angleBetweenVectors(attitude.rotate(Vector(0, 0, 1)), Vector(0, 0, 1));
|
||||
setLED(angle < radians(15));
|
||||
}
|
||||
|
||||
@@ -8,35 +8,36 @@
|
||||
|
||||
#define SERIAL_BAUDRATE 115200
|
||||
|
||||
#define WIFI_ENABLED 0
|
||||
#define WIFI_ENABLED 1
|
||||
|
||||
#define RC_CHANNELS 6
|
||||
#define RC_CHANNELS 16
|
||||
#define RC_CHANNEL_ROLL 0
|
||||
#define RC_CHANNEL_PITCH 1
|
||||
#define RC_CHANNEL_THROTTLE 2
|
||||
#define RC_CHANNEL_YAW 3
|
||||
#define RC_CHANNEL_PITCH 1
|
||||
#define RC_CHANNEL_ROLL 0
|
||||
#define RC_CHANNEL_AUX 4
|
||||
#define RC_CHANNEL_ARMED 4
|
||||
#define RC_CHANNEL_MODE 5
|
||||
|
||||
#define MOTOR_REAR_LEFT 0
|
||||
#define MOTOR_FRONT_LEFT 3
|
||||
#define MOTOR_FRONT_RIGHT 2
|
||||
#define MOTOR_REAR_RIGHT 1
|
||||
#define MOTOR_FRONT_RIGHT 2
|
||||
#define MOTOR_FRONT_LEFT 3
|
||||
|
||||
float t = NAN; // current step time, s
|
||||
float dt; // time delta from previous step, s
|
||||
float loopFreq; // loop frequency, Hz
|
||||
uint16_t channels[16]; // raw rc channels
|
||||
int16_t channels[RC_CHANNELS]; // raw rc channels
|
||||
float controls[RC_CHANNELS]; // normalized controls in range [-1..1] ([0..1] for throttle)
|
||||
Vector rates; // angular rates, rad/s
|
||||
Vector gyro; // gyroscope data
|
||||
Vector acc; // accelerometer data, m/s/s
|
||||
Vector rates; // filtered angular rates, rad/s
|
||||
Quaternion attitude; // estimated attitude
|
||||
float motors[4]; // normalized motors thrust in range [-1..1]
|
||||
|
||||
void setup()
|
||||
{
|
||||
void setup() {
|
||||
Serial.begin(SERIAL_BAUDRATE);
|
||||
Serial.println("Initializing flix");
|
||||
disableBrownOut();
|
||||
setupLED();
|
||||
setupMotors();
|
||||
setLED(true);
|
||||
@@ -50,10 +51,8 @@ void setup()
|
||||
Serial.println("Initializing complete");
|
||||
}
|
||||
|
||||
void loop()
|
||||
{
|
||||
if (!readIMU()) return;
|
||||
|
||||
void loop() {
|
||||
readIMU();
|
||||
step();
|
||||
readRC();
|
||||
estimate();
|
||||
@@ -61,7 +60,7 @@ void loop()
|
||||
sendMotors();
|
||||
parseInput();
|
||||
#if WIFI_ENABLED == 1
|
||||
sendMavlink();
|
||||
processMavlink();
|
||||
#endif
|
||||
logData();
|
||||
}
|
||||
|
||||
167
flix/imu.ino
@@ -2,104 +2,123 @@
|
||||
// Repository: https://github.com/okalachev/flix
|
||||
|
||||
// Work with the IMU sensor
|
||||
// IMU is oriented FLU (front-left-up) style.
|
||||
// In case of FRD (front-right-down) orientation of the IMU, use this code:
|
||||
// https://gist.github.com/okalachev/713db47e31bce643dbbc9539d166ce98.
|
||||
|
||||
#include <SPI.h>
|
||||
#include <MPU9250.h>
|
||||
|
||||
#define IMU_CS_PIN 4 // chip-select pin for IMU SPI connection
|
||||
#define CALIBRATE_GYRO_ON_START true
|
||||
#define ONE_G 9.80665
|
||||
|
||||
MPU9250 IMU(SPI, IMU_CS_PIN);
|
||||
// NOTE: use 'ca' command to calibrate the accelerometer and put the values here
|
||||
Vector accBias(0, 0, 0);
|
||||
Vector accScale(1, 1, 1);
|
||||
|
||||
void setupIMU()
|
||||
{
|
||||
MPU9250 IMU(SPI);
|
||||
Vector gyroBias;
|
||||
|
||||
void setupIMU() {
|
||||
Serial.println("Setup IMU");
|
||||
|
||||
auto status = IMU.begin();
|
||||
if (status < 0) {
|
||||
bool status = IMU.begin();
|
||||
if (!status) {
|
||||
while (true) {
|
||||
Serial.printf("IMU begin error: %d\n", status);
|
||||
Serial.println("IMU begin error");
|
||||
delay(1000);
|
||||
}
|
||||
}
|
||||
calibrateGyro();
|
||||
}
|
||||
|
||||
if (CALIBRATE_GYRO_ON_START) {
|
||||
calibrateGyro();
|
||||
} else {
|
||||
loadGyroCal();
|
||||
}
|
||||
|
||||
loadAccelCal();
|
||||
|
||||
void configureIMU() {
|
||||
IMU.setAccelRange(IMU.ACCEL_RANGE_4G);
|
||||
IMU.setGyroRange(IMU.GYRO_RANGE_2000DPS);
|
||||
IMU.setDlpfBandwidth(IMU.DLPF_BANDWIDTH_184HZ);
|
||||
IMU.setSrd(0); // set sample rate to 1000 Hz
|
||||
// NOTE: very important, without the above the rate would be terrible 50 Hz
|
||||
}
|
||||
|
||||
bool readIMU()
|
||||
{
|
||||
if (IMU.readSensor() < 0) {
|
||||
Serial.println("IMU read error");
|
||||
return false;
|
||||
}
|
||||
|
||||
auto lastRates = rates;
|
||||
|
||||
rates.x = IMU.getGyroX_rads();
|
||||
rates.y = IMU.getGyroY_rads();
|
||||
rates.z = IMU.getGyroZ_rads();
|
||||
acc.x = IMU.getAccelX_mss();
|
||||
acc.y = IMU.getAccelY_mss();
|
||||
acc.z = IMU.getAccelZ_mss();
|
||||
|
||||
return rates != lastRates;
|
||||
void readIMU() {
|
||||
IMU.waitForData();
|
||||
IMU.getGyro(gyro.x, gyro.y, gyro.z);
|
||||
IMU.getAccel(acc.x, acc.y, acc.z);
|
||||
// apply scale and bias
|
||||
acc = (acc - accBias) / accScale;
|
||||
gyro = gyro - gyroBias;
|
||||
}
|
||||
|
||||
void calibrateGyro()
|
||||
{
|
||||
void calibrateGyro() {
|
||||
const int samples = 1000;
|
||||
Serial.println("Calibrating gyro, stand still");
|
||||
delay(500);
|
||||
int status = IMU.calibrateGyro();
|
||||
Serial.printf("Calibration status: %d\n", status);
|
||||
IMU.setSrd(0);
|
||||
IMU.setGyroRange(IMU.GYRO_RANGE_250DPS); // the most sensitive mode
|
||||
|
||||
gyroBias = Vector(0, 0, 0);
|
||||
for (int i = 0; i < samples; i++) {
|
||||
IMU.waitForData();
|
||||
IMU.getGyro(gyro.x, gyro.y, gyro.z);
|
||||
gyroBias = gyroBias + gyro;
|
||||
}
|
||||
gyroBias = gyroBias / samples;
|
||||
|
||||
printIMUCal();
|
||||
configureIMU();
|
||||
}
|
||||
|
||||
void calibrateAccel()
|
||||
{
|
||||
Serial.println("Cal accel: place level"); delay(3000);
|
||||
IMU.calibrateAccel();
|
||||
Serial.println("Cal accel: place nose up"); delay(3000);
|
||||
IMU.calibrateAccel();
|
||||
Serial.println("Cal accel: place nose down"); delay(3000);
|
||||
IMU.calibrateAccel();
|
||||
Serial.println("Cal accel: place on right side"); delay(3000);
|
||||
IMU.calibrateAccel();
|
||||
Serial.println("Cal accel: place on left side"); delay(3000);
|
||||
IMU.calibrateAccel();
|
||||
Serial.println("Cal accel: upside down"); delay(300);
|
||||
IMU.calibrateAccel();
|
||||
void calibrateAccel() {
|
||||
Serial.println("Calibrating accelerometer");
|
||||
IMU.setAccelRange(IMU.ACCEL_RANGE_2G); // the most sensitive mode
|
||||
IMU.setDlpfBandwidth(IMU.DLPF_BANDWIDTH_20HZ);
|
||||
IMU.setSrd(19);
|
||||
|
||||
Serial.setTimeout(60000);
|
||||
Serial.print("Place level [enter] "); Serial.readStringUntil('\n');
|
||||
calibrateAccelOnce();
|
||||
Serial.print("Place nose up [enter] "); Serial.readStringUntil('\n');
|
||||
calibrateAccelOnce();
|
||||
Serial.print("Place nose down [enter] "); Serial.readStringUntil('\n');
|
||||
calibrateAccelOnce();
|
||||
Serial.print("Place on right side [enter] "); Serial.readStringUntil('\n');
|
||||
calibrateAccelOnce();
|
||||
Serial.print("Place on left side [enter] "); Serial.readStringUntil('\n');
|
||||
calibrateAccelOnce();
|
||||
Serial.print("Place upside down [enter] "); Serial.readStringUntil('\n');
|
||||
calibrateAccelOnce();
|
||||
|
||||
printIMUCal();
|
||||
configureIMU();
|
||||
}
|
||||
|
||||
void loadAccelCal()
|
||||
{
|
||||
// NOTE: this should be changed to the actual values
|
||||
IMU.setAccelCalX(-0.0048542023, 1.0008112192);
|
||||
IMU.setAccelCalY(0.0521845818, 0.9985780716);
|
||||
IMU.setAccelCalZ(0.5754694939, 1.0045746565);
|
||||
void calibrateAccelOnce() {
|
||||
const int samples = 100;
|
||||
static Vector accMax(-INFINITY, -INFINITY, -INFINITY);
|
||||
static Vector accMin(INFINITY, INFINITY, INFINITY);
|
||||
|
||||
// Compute the average of the accelerometer readings
|
||||
acc = Vector(0, 0, 0);
|
||||
for (int i = 0; i < samples; i++) {
|
||||
IMU.waitForData();
|
||||
Vector sample;
|
||||
IMU.getAccel(sample.x, sample.y, sample.z);
|
||||
acc = acc + sample;
|
||||
}
|
||||
acc = acc / samples;
|
||||
|
||||
// Update the maximum and minimum values
|
||||
if (acc.x > accMax.x) accMax.x = acc.x;
|
||||
if (acc.y > accMax.y) accMax.y = acc.y;
|
||||
if (acc.z > accMax.z) accMax.z = acc.z;
|
||||
if (acc.x < accMin.x) accMin.x = acc.x;
|
||||
if (acc.y < accMin.y) accMin.y = acc.y;
|
||||
if (acc.z < accMin.z) accMin.z = acc.z;
|
||||
Serial.printf("acc %f %f %f\n", acc.x, acc.y, acc.z);
|
||||
Serial.printf("max %f %f %f\n", accMax.x, accMax.y, accMax.z);
|
||||
Serial.printf("min %f %f %f\n", accMin.x, accMin.y, accMin.z);
|
||||
// Compute scale and bias
|
||||
accScale = (accMax - accMin) / 2 / ONE_G;
|
||||
accBias = (accMax + accMin) / 2;
|
||||
}
|
||||
|
||||
void loadGyroCal()
|
||||
{
|
||||
// NOTE: this should be changed to the actual values
|
||||
IMU.setGyroBiasX_rads(-0.0185128022);
|
||||
IMU.setGyroBiasY_rads(-0.0262369743);
|
||||
IMU.setGyroBiasZ_rads(0.0163032326);
|
||||
}
|
||||
|
||||
void printIMUCal()
|
||||
{
|
||||
Serial.printf("gyro bias: %f %f %f\n", IMU.getGyroBiasX_rads(), IMU.getGyroBiasY_rads(), IMU.getGyroBiasZ_rads());
|
||||
Serial.printf("accel bias: %f %f %f\n", IMU.getAccelBiasX_mss(), IMU.getAccelBiasY_mss(), IMU.getAccelBiasZ_mss());
|
||||
Serial.printf("accel scale: %f %f %f\n", IMU.getAccelScaleFactorX(), IMU.getAccelScaleFactorY(), IMU.getAccelScaleFactorZ());
|
||||
void printIMUCal() {
|
||||
Serial.printf("gyro bias: %f %f %f\n", gyroBias.x, gyroBias.y, gyroBias.z);
|
||||
Serial.printf("accel bias: %f %f %f\n", accBias.x, accBias.y, accBias.z);
|
||||
Serial.printf("accel scale: %f %f %f\n", accScale.x, accScale.y, accScale.z);
|
||||
}
|
||||
|
||||
13
flix/led.ino
@@ -5,17 +5,18 @@
|
||||
|
||||
#define BLINK_PERIOD 500000
|
||||
|
||||
void setupLED()
|
||||
{
|
||||
#ifndef LED_BUILTIN
|
||||
#define LED_BUILTIN 2 // for ESP32 Dev Module
|
||||
#endif
|
||||
|
||||
void setupLED() {
|
||||
pinMode(LED_BUILTIN, OUTPUT);
|
||||
}
|
||||
|
||||
void setLED(bool on)
|
||||
{
|
||||
void setLED(bool on) {
|
||||
digitalWrite(LED_BUILTIN, on ? HIGH : LOW);
|
||||
}
|
||||
|
||||
void blinkLED()
|
||||
{
|
||||
void blinkLED() {
|
||||
setLED(micros() / BLINK_PERIOD % 2);
|
||||
}
|
||||
|
||||
@@ -12,8 +12,7 @@
|
||||
float logBuffer[LOG_SIZE][LOG_COLUMNS]; // * 4 (float)
|
||||
int logPointer = 0;
|
||||
|
||||
void logData()
|
||||
{
|
||||
void logData() {
|
||||
if (!armed) return;
|
||||
|
||||
static float logTime = 0;
|
||||
@@ -41,11 +40,11 @@ void logData()
|
||||
}
|
||||
}
|
||||
|
||||
void dumpLog()
|
||||
{
|
||||
void dumpLog() {
|
||||
Serial.printf("t,rates.x,rates.y,rates.z,ratesTarget.x,ratesTarget.y,ratesTarget.z,"
|
||||
"attitude.x,attitude.y,attitude.z,attitudeTarget.x,attitudeTarget.y,attitudeTarget.z,thrustTarget\n");
|
||||
for (int i = 0; i < LOG_SIZE; i++) {
|
||||
if (logBuffer[i][0] == 0) continue; // skip empty records
|
||||
for (int j = 0; j < LOG_COLUMNS - 1; j++) {
|
||||
Serial.printf("%f,", logBuffer[i][j]);
|
||||
}
|
||||
|
||||
12
flix/lpf.h
@@ -6,16 +6,14 @@
|
||||
#pragma once
|
||||
|
||||
template <typename T> // Using template to make the filter usable for scalar and vector values
|
||||
class LowPassFilter
|
||||
{
|
||||
class LowPassFilter {
|
||||
public:
|
||||
float alpha; // smoothing constant, 1 means filter disabled
|
||||
T output;
|
||||
|
||||
LowPassFilter(float alpha): alpha(alpha) {};
|
||||
|
||||
T update(const T input)
|
||||
{
|
||||
T update(const T input) {
|
||||
if (alpha == 1) { // filter disabled
|
||||
return input;
|
||||
}
|
||||
@@ -27,13 +25,11 @@ public:
|
||||
return output = output * (1 - alpha) + input * alpha;
|
||||
}
|
||||
|
||||
void setCutOffFrequency(float cutOffFreq, float dt)
|
||||
{
|
||||
void setCutOffFrequency(float cutOffFreq, float dt) {
|
||||
alpha = 1 - exp(-2 * PI * cutOffFreq * dt);
|
||||
}
|
||||
|
||||
void reset()
|
||||
{
|
||||
void reset() {
|
||||
initialized = false;
|
||||
}
|
||||
|
||||
|
||||
@@ -5,14 +5,20 @@
|
||||
|
||||
#if WIFI_ENABLED == 1
|
||||
|
||||
#include "mavlink/common/mavlink.h"
|
||||
#include <MAVLink.h>
|
||||
|
||||
#define SYSTEM_ID 1
|
||||
#define PERIOD_SLOW 1.0
|
||||
#define PERIOD_FAST 0.1
|
||||
#define MAVLINK_CONTROL_SCALE 0.7f
|
||||
#define MAVLINK_CONTROL_YAW_DEAD_ZONE 0.1f
|
||||
|
||||
void sendMavlink()
|
||||
{
|
||||
void processMavlink() {
|
||||
sendMavlink();
|
||||
receiveMavlink();
|
||||
}
|
||||
|
||||
void sendMavlink() {
|
||||
static float lastSlow = 0;
|
||||
static float lastFast = 0;
|
||||
|
||||
@@ -23,7 +29,7 @@ void sendMavlink()
|
||||
lastSlow = t;
|
||||
|
||||
mavlink_msg_heartbeat_pack(SYSTEM_ID, MAV_COMP_ID_AUTOPILOT1, &msg, MAV_TYPE_QUADROTOR,
|
||||
MAV_AUTOPILOT_GENERIC, MAV_MODE_FLAG_MANUAL_INPUT_ENABLED | armed ? MAV_MODE_FLAG_SAFETY_ARMED : 0,
|
||||
MAV_AUTOPILOT_GENERIC, MAV_MODE_FLAG_MANUAL_INPUT_ENABLED | (armed ? MAV_MODE_FLAG_SAFETY_ARMED : 0),
|
||||
0, MAV_STATE_STANDBY);
|
||||
sendMessage(&msg);
|
||||
}
|
||||
@@ -32,34 +38,69 @@ void sendMavlink()
|
||||
lastFast = t;
|
||||
|
||||
const float zeroQuat[] = {0, 0, 0, 0};
|
||||
Quaternion attitudeFRD = FLU2FRD(attitude); // MAVLink uses FRD coordinate system
|
||||
mavlink_msg_attitude_quaternion_pack(SYSTEM_ID, MAV_COMP_ID_AUTOPILOT1, &msg,
|
||||
time, attitude.w, attitude.x, attitude.y, attitude.z, rates.x, rates.y, rates.z, zeroQuat);
|
||||
time, attitudeFRD.w, attitudeFRD.x, attitudeFRD.y, attitudeFRD.z, rates.x, rates.y, rates.z, zeroQuat);
|
||||
sendMessage(&msg);
|
||||
|
||||
mavlink_msg_rc_channels_scaled_pack(SYSTEM_ID, MAV_COMP_ID_AUTOPILOT1, &msg, time, 0,
|
||||
controls[0] * 10000, controls[1] * 10000, controls[2] * 10000,
|
||||
controls[3] * 10000, controls[4] * 10000, controls[5] * 10000,
|
||||
UINT16_MAX, UINT16_MAX, 255);
|
||||
INT16_MAX, INT16_MAX, UINT8_MAX);
|
||||
sendMessage(&msg);
|
||||
|
||||
float actuator[32];
|
||||
memcpy(motors, actuator, 4 * sizeof(float));
|
||||
memcpy(actuator, motors, sizeof(motors));
|
||||
mavlink_msg_actuator_output_status_pack(SYSTEM_ID, MAV_COMP_ID_AUTOPILOT1, &msg, time, 4, actuator);
|
||||
sendMessage(&msg);
|
||||
|
||||
mavlink_msg_scaled_imu_pack(SYSTEM_ID, MAV_COMP_ID_AUTOPILOT1, &msg, time,
|
||||
acc.x * 1000, acc.y * 1000, acc.z * 1000,
|
||||
rates.x * 1000, rates.y * 1000, rates.z * 1000,
|
||||
gyro.x * 1000, gyro.y * 1000, gyro.z * 1000,
|
||||
0, 0, 0, 0);
|
||||
sendMessage(&msg);
|
||||
}
|
||||
}
|
||||
|
||||
inline void sendMessage(const void *msg)
|
||||
{
|
||||
void sendMessage(const void *msg) {
|
||||
uint8_t buf[MAVLINK_MAX_PACKET_LEN];
|
||||
uint16_t len = mavlink_msg_to_send_buffer(buf, (mavlink_message_t *)msg);
|
||||
int len = mavlink_msg_to_send_buffer(buf, (mavlink_message_t *)msg);
|
||||
sendWiFi(buf, len);
|
||||
}
|
||||
|
||||
void receiveMavlink() {
|
||||
uint8_t buf[MAVLINK_MAX_PACKET_LEN];
|
||||
int len = receiveWiFi(buf, MAVLINK_MAX_PACKET_LEN);
|
||||
|
||||
// New packet, parse it
|
||||
mavlink_message_t msg;
|
||||
mavlink_status_t status;
|
||||
for (int i = 0; i < len; i++) {
|
||||
if (mavlink_parse_char(MAVLINK_COMM_0, buf[i], &msg, &status)) {
|
||||
handleMavlink(&msg);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void handleMavlink(const void *_msg) {
|
||||
mavlink_message_t *msg = (mavlink_message_t *)_msg;
|
||||
if (msg->msgid == MAVLINK_MSG_ID_MANUAL_CONTROL) {
|
||||
mavlink_manual_control_t manualControl;
|
||||
mavlink_msg_manual_control_decode(msg, &manualControl);
|
||||
controls[RC_CHANNEL_THROTTLE] = manualControl.z / 1000.0f;
|
||||
controls[RC_CHANNEL_PITCH] = manualControl.x / 1000.0f * MAVLINK_CONTROL_SCALE;
|
||||
controls[RC_CHANNEL_ROLL] = manualControl.y / 1000.0f * MAVLINK_CONTROL_SCALE;
|
||||
controls[RC_CHANNEL_YAW] = manualControl.r / 1000.0f * MAVLINK_CONTROL_SCALE;
|
||||
controls[RC_CHANNEL_MODE] = 1; // STAB mode
|
||||
controls[RC_CHANNEL_ARMED] = 1; // armed
|
||||
|
||||
if (abs(controls[RC_CHANNEL_YAW]) < MAVLINK_CONTROL_YAW_DEAD_ZONE) controls[RC_CHANNEL_YAW] = 0;
|
||||
}
|
||||
}
|
||||
|
||||
// Convert Forward-Left-Up to Forward-Right-Down quaternion
|
||||
inline Quaternion FLU2FRD(const Quaternion &q) {
|
||||
return Quaternion(q.w, q.x, -q.y, -q.z);
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
@@ -1,75 +1,50 @@
|
||||
// Copyright (c) 2023 Oleg Kalachev <okalachev@gmail.com>
|
||||
// Repository: https://github.com/okalachev/flix
|
||||
|
||||
// Motors output control
|
||||
// Motors output control using MOSFETs
|
||||
// In case of using ESC, use this version of the code: https://gist.github.com/okalachev/8871d3a94b6b6c0a298f41a4edd34c61.
|
||||
// Motor: 8520 3.7V
|
||||
// ESC: KINGDUO Micro Mini 4A 1S Brushed Esc 3.6-6V
|
||||
|
||||
#define MOTOR_0_PIN 12
|
||||
#define MOTOR_1_PIN 13
|
||||
#define MOTOR_2_PIN 14
|
||||
#define MOTOR_3_PIN 15
|
||||
#define MOTOR_0_PIN 12 // rear left
|
||||
#define MOTOR_1_PIN 13 // rear right
|
||||
#define MOTOR_2_PIN 14 // front right
|
||||
#define MOTOR_3_PIN 15 // front left
|
||||
|
||||
#define PWM_FREQUENCY 200
|
||||
#define PWM_RESOLUTION 8
|
||||
|
||||
#define PWM_NEUTRAL 1500
|
||||
|
||||
const uint16_t pwmMin[] = {1600, 1600, 1600, 1600};
|
||||
const uint16_t pwmMax[] = {2300, 2300, 2300, 2300};
|
||||
const uint16_t pwmReverseMin[] = {1390, 1440, 1440, 1440};
|
||||
const uint16_t pwmReverseMax[] = {1100, 1100, 1100, 1100};
|
||||
|
||||
void setupMotors() {
|
||||
Serial.println("Setup Motors");
|
||||
|
||||
// configure PWM channels
|
||||
ledcSetup(0, PWM_FREQUENCY, PWM_RESOLUTION);
|
||||
ledcSetup(1, PWM_FREQUENCY, PWM_RESOLUTION);
|
||||
ledcSetup(2, PWM_FREQUENCY, PWM_RESOLUTION);
|
||||
ledcSetup(3, PWM_FREQUENCY, PWM_RESOLUTION);
|
||||
|
||||
// attach channels to motor pins
|
||||
ledcAttachPin(MOTOR_0_PIN, 0);
|
||||
ledcAttachPin(MOTOR_1_PIN, 1);
|
||||
ledcAttachPin(MOTOR_2_PIN, 2);
|
||||
ledcAttachPin(MOTOR_3_PIN, 3);
|
||||
// configure pins
|
||||
ledcAttach(MOTOR_0_PIN, PWM_FREQUENCY, PWM_RESOLUTION);
|
||||
ledcAttach(MOTOR_1_PIN, PWM_FREQUENCY, PWM_RESOLUTION);
|
||||
ledcAttach(MOTOR_2_PIN, PWM_FREQUENCY, PWM_RESOLUTION);
|
||||
ledcAttach(MOTOR_3_PIN, PWM_FREQUENCY, PWM_RESOLUTION);
|
||||
|
||||
sendMotors();
|
||||
Serial.println("Motors initialized");
|
||||
}
|
||||
|
||||
uint16_t getPWM(float val, int n)
|
||||
{
|
||||
if (val == 0) {
|
||||
return PWM_NEUTRAL;
|
||||
} else if (val > 0) {
|
||||
return mapff(val, 0, 1, pwmMin[n], pwmMax[n]);
|
||||
} else {
|
||||
return mapff(val, 0, -1, pwmReverseMin[n], pwmReverseMax[n]);
|
||||
}
|
||||
uint8_t signalToDutyCycle(float control) {
|
||||
float duty = mapff(control, 0, 1, 0, (1 << PWM_RESOLUTION) - 1);
|
||||
return round(constrain(duty, 0, (1 << PWM_RESOLUTION) - 1));
|
||||
}
|
||||
|
||||
uint8_t pwmToDutyCycle(uint16_t pwm) {
|
||||
return map(pwm, 0, 1000000 / PWM_FREQUENCY, 0, (1 << PWM_RESOLUTION) - 1);
|
||||
void sendMotors() {
|
||||
ledcWrite(MOTOR_0_PIN, signalToDutyCycle(motors[0]));
|
||||
ledcWrite(MOTOR_1_PIN, signalToDutyCycle(motors[1]));
|
||||
ledcWrite(MOTOR_2_PIN, signalToDutyCycle(motors[2]));
|
||||
ledcWrite(MOTOR_3_PIN, signalToDutyCycle(motors[3]));
|
||||
}
|
||||
|
||||
void sendMotors()
|
||||
{
|
||||
ledcWrite(0, pwmToDutyCycle(getPWM(motors[0], 0)));
|
||||
ledcWrite(1, pwmToDutyCycle(getPWM(motors[1], 1)));
|
||||
ledcWrite(2, pwmToDutyCycle(getPWM(motors[2], 2)));
|
||||
ledcWrite(3, pwmToDutyCycle(getPWM(motors[3], 3)));
|
||||
}
|
||||
|
||||
void fullMotorTest(int n, bool reverse)
|
||||
{
|
||||
void fullMotorTest(int n) {
|
||||
printf("Full test for motor %d\n", n);
|
||||
for (int pwm = PWM_NEUTRAL; pwm <= 2300 && pwm >= 700; pwm += reverse ? -100 : 100) {
|
||||
printf("Motor %d: %d\n", n, pwm);
|
||||
ledcWrite(n, pwmToDutyCycle(pwm));
|
||||
for (float signal = 0; signal <= 1; signal += 0.1) {
|
||||
printf("Motor %d: %f\n", n, signal);
|
||||
ledcWrite(n, signalToDutyCycle(signal));
|
||||
delay(3000);
|
||||
}
|
||||
printf("Motor %d: %d\n", n, PWM_NEUTRAL);
|
||||
ledcWrite(n, pwmToDutyCycle(PWM_NEUTRAL));
|
||||
printf("Motor %d: %f\n", n, 0);
|
||||
ledcWrite(n, signalToDutyCycle(0));
|
||||
}
|
||||
|
||||
@@ -7,8 +7,7 @@
|
||||
|
||||
#include "lpf.h"
|
||||
|
||||
class PID
|
||||
{
|
||||
class PID {
|
||||
public:
|
||||
float p = 0;
|
||||
float i = 0;
|
||||
@@ -22,8 +21,7 @@ public:
|
||||
|
||||
PID(float p, float i, float d, float windup = 0, float dAlpha = 1) : p(p), i(i), d(d), windup(windup), lpf(dAlpha) {};
|
||||
|
||||
float update(float error, float dt)
|
||||
{
|
||||
float update(float error, float dt) {
|
||||
integral += error * dt;
|
||||
|
||||
if (isfinite(prevError) && dt > 0) {
|
||||
@@ -39,8 +37,7 @@ public:
|
||||
return p * error + constrain(i * integral, -windup, windup) + d * derivative; // PID
|
||||
}
|
||||
|
||||
void reset()
|
||||
{
|
||||
void reset() {
|
||||
prevError = NAN;
|
||||
integral = 0;
|
||||
derivative = 0;
|
||||
|
||||
@@ -15,8 +15,7 @@ public:
|
||||
|
||||
Quaternion(float w, float x, float y, float z): w(w), x(x), y(y), z(z) {};
|
||||
|
||||
static Quaternion fromAxisAngle(float a, float b, float c, float angle)
|
||||
{
|
||||
static Quaternion fromAxisAngle(float a, float b, float c, float angle) {
|
||||
float halfAngle = angle * 0.5;
|
||||
float sin2 = sin(halfAngle);
|
||||
float cos2 = cos(halfAngle);
|
||||
@@ -24,27 +23,20 @@ public:
|
||||
return Quaternion(cos2, a * sinNorm, b * sinNorm, c * sinNorm);
|
||||
}
|
||||
|
||||
static Quaternion fromAngularRates(float x, float y, float z)
|
||||
{
|
||||
return Quaternion::fromAxisAngle(x, y, z, sqrt(x * x + y * y + z * z));
|
||||
}
|
||||
|
||||
static Quaternion fromAngularRates(const Vector& rates)
|
||||
{
|
||||
static Quaternion fromAngularRates(const Vector& rates) {
|
||||
if (rates.zero()) {
|
||||
return Quaternion();
|
||||
}
|
||||
return Quaternion::fromAxisAngle(rates.x, rates.y, rates.z, rates.norm());
|
||||
}
|
||||
|
||||
static Quaternion fromEulerZYX(float x, float y, float z)
|
||||
{
|
||||
float cx = cos(x / 2);
|
||||
float cy = cos(y / 2);
|
||||
float cz = cos(z / 2);
|
||||
float sx = sin(x / 2);
|
||||
float sy = sin(y / 2);
|
||||
float sz = sin(z / 2);
|
||||
static Quaternion fromEulerZYX(const Vector& euler) {
|
||||
float cx = cos(euler.x / 2);
|
||||
float cy = cos(euler.y / 2);
|
||||
float cz = cos(euler.z / 2);
|
||||
float sx = sin(euler.x / 2);
|
||||
float sy = sin(euler.y / 2);
|
||||
float sz = sin(euler.z / 2);
|
||||
|
||||
return Quaternion(
|
||||
cx * cy * cz + sx * sy * sz,
|
||||
@@ -53,8 +45,7 @@ public:
|
||||
cx * cy * sz - sx * sy * cz);
|
||||
}
|
||||
|
||||
static Quaternion fromBetweenVectors(Vector u, Vector v)
|
||||
{
|
||||
static Quaternion fromBetweenVectors(Vector u, Vector v) {
|
||||
float dot = u.x * v.x + u.y * v.y + u.z * v.z;
|
||||
float w1 = u.y * v.z - u.z * v.y;
|
||||
float w2 = u.z * v.x - u.x * v.z;
|
||||
@@ -69,33 +60,46 @@ public:
|
||||
return ret;
|
||||
}
|
||||
|
||||
void toAxisAngle(float& a, float& b, float& c, float& angle)
|
||||
{
|
||||
void toAxisAngle(float& a, float& b, float& c, float& angle) {
|
||||
angle = acos(w) * 2;
|
||||
a = x / sin(angle / 2);
|
||||
b = y / sin(angle / 2);
|
||||
c = z / sin(angle / 2);
|
||||
}
|
||||
|
||||
Vector toEulerZYX() const
|
||||
{
|
||||
return Vector(
|
||||
atan2(2 * (w * x + y * z), 1 - 2 * (x * x + y * y)),
|
||||
asin(2 * (w * y - z * x)),
|
||||
atan2(2 * (w * z + x * y), 1 - 2 * (y * y + z * z)));
|
||||
Vector toEulerZYX() const {
|
||||
// https://github.com/ros/geometry2/blob/589caf083cae9d8fae7effdb910454b4681b9ec1/tf2/include/tf2/impl/utils.h#L87
|
||||
Vector euler;
|
||||
float sqx = x * x;
|
||||
float sqy = y * y;
|
||||
float sqz = z * z;
|
||||
float sqw = w * w;
|
||||
// Cases derived from https://orbitalstation.wordpress.com/tag/quaternion/
|
||||
float sarg = -2 * (x * z - w * y) / (sqx + sqy + sqz + sqw);
|
||||
if (sarg <= -0.99999) {
|
||||
euler.x = 0;
|
||||
euler.y = -0.5 * PI;
|
||||
euler.z = -2 * atan2(y, x);
|
||||
} else if (sarg >= 0.99999) {
|
||||
euler.x = 0;
|
||||
euler.y = 0.5 * PI;
|
||||
euler.z = 2 * atan2(y, x);
|
||||
} else {
|
||||
euler.x = atan2(2 * (y * z + w * x), sqw - sqx - sqy + sqz);
|
||||
euler.y = asin(sarg);
|
||||
euler.z = atan2(2 * (x * y + w * z), sqw + sqx - sqy - sqz);
|
||||
}
|
||||
return euler;
|
||||
}
|
||||
|
||||
float getYaw() const
|
||||
{
|
||||
float getYaw() const {
|
||||
// https://github.com/ros/geometry2/blob/589caf083cae9d8fae7effdb910454b4681b9ec1/tf2/include/tf2/impl/utils.h#L122
|
||||
float yaw;
|
||||
float sqx = x * x;
|
||||
float sqy = y * y;
|
||||
float sqz = z * z;
|
||||
float sqw = w * w;
|
||||
|
||||
double sarg = -2 * (x * z - w * y) / (sqx + sqy + sqz + sqw);
|
||||
|
||||
if (sarg <= -0.99999) {
|
||||
yaw = -2 * atan2(y, x);
|
||||
} else if (sarg >= 0.99999) {
|
||||
@@ -106,15 +110,14 @@ public:
|
||||
return yaw;
|
||||
}
|
||||
|
||||
void setYaw(float yaw)
|
||||
{
|
||||
void setYaw(float yaw) {
|
||||
// TODO: optimize?
|
||||
Vector euler = toEulerZYX();
|
||||
(*this) = Quaternion::fromEulerZYX(euler.x, euler.y, yaw);
|
||||
euler.z = yaw;
|
||||
(*this) = Quaternion::fromEulerZYX(euler);
|
||||
}
|
||||
|
||||
Quaternion& operator *= (const Quaternion& q)
|
||||
{
|
||||
Quaternion& operator *= (const Quaternion& q) {
|
||||
Quaternion ret(
|
||||
w * q.w - x * q.x - y * q.y - z * q.z,
|
||||
w * q.x + x * q.w + y * q.z - z * q.y,
|
||||
@@ -123,8 +126,7 @@ public:
|
||||
return (*this = ret);
|
||||
}
|
||||
|
||||
Quaternion operator * (const Quaternion& q)
|
||||
{
|
||||
Quaternion operator * (const Quaternion& q) {
|
||||
return Quaternion(
|
||||
w * q.w - x * q.x - y * q.y - z * q.z,
|
||||
w * q.x + x * q.w + y * q.z - z * q.y,
|
||||
@@ -132,8 +134,7 @@ public:
|
||||
w * q.z + z * q.w + x * q.y - y * q.x);
|
||||
}
|
||||
|
||||
Quaternion inversed() const
|
||||
{
|
||||
Quaternion inversed() const {
|
||||
float normSqInv = 1 / (w * w + x * x + y * y + z * z);
|
||||
return Quaternion(
|
||||
w * normSqInv,
|
||||
@@ -142,13 +143,11 @@ public:
|
||||
-z * normSqInv);
|
||||
}
|
||||
|
||||
float norm() const
|
||||
{
|
||||
float norm() const {
|
||||
return sqrt(w * w + x * x + y * y + z * z);
|
||||
}
|
||||
|
||||
void normalize()
|
||||
{
|
||||
void normalize() {
|
||||
float n = norm();
|
||||
w /= n;
|
||||
x /= n;
|
||||
@@ -156,27 +155,24 @@ public:
|
||||
z /= n;
|
||||
}
|
||||
|
||||
Vector conjugate(const Vector& v)
|
||||
{
|
||||
Vector conjugate(const Vector& v) {
|
||||
Quaternion qv(0, v.x, v.y, v.z);
|
||||
Quaternion res = (*this) * qv * inversed();
|
||||
return Vector(res.x, res.y, res.z);
|
||||
}
|
||||
|
||||
Vector conjugateInversed(const Vector& v)
|
||||
{
|
||||
Vector conjugateInversed(const Vector& v) {
|
||||
Quaternion qv(0, v.x, v.y, v.z);
|
||||
Quaternion res = inversed() * qv * (*this);
|
||||
return Vector(res.x, res.y, res.z);
|
||||
}
|
||||
|
||||
inline Vector rotate(const Vector& v)
|
||||
{
|
||||
// Rotate vector by quaternion
|
||||
inline Vector rotate(const Vector& v) {
|
||||
return conjugateInversed(v);
|
||||
}
|
||||
|
||||
inline bool finite() const
|
||||
{
|
||||
inline bool finite() const {
|
||||
return isfinite(w) && isfinite(x) && isfinite(y) && isfinite(z);
|
||||
}
|
||||
|
||||
|
||||
45
flix/rc.ino
@@ -5,29 +5,50 @@
|
||||
|
||||
#include <SBUS.h>
|
||||
|
||||
const uint16_t channelNeutral[] = {995, 883, 200, 972, 512, 512};
|
||||
const uint16_t channelMax[] = {1651, 1540, 1713, 1630, 1472, 1472};
|
||||
// NOTE: use 'cr' command to calibrate the RC and put the values here
|
||||
int channelNeutral[] = {995, 883, 200, 972, 512, 512, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
|
||||
int channelMax[] = {1651, 1540, 1713, 1630, 1472, 1472, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
|
||||
|
||||
SBUS RC(Serial2);
|
||||
SBUS RC(Serial2, 16, 17); // NOTE: remove pin numbers (16, 17) if you use the new default pins for Serial2 (4, 25)
|
||||
|
||||
void setupRC()
|
||||
{
|
||||
void setupRC() {
|
||||
Serial.println("Setup RC");
|
||||
RC.begin();
|
||||
}
|
||||
|
||||
void readRC()
|
||||
{
|
||||
bool failSafe, lostFrame;
|
||||
if (RC.read(channels, &failSafe, &lostFrame)) {
|
||||
if (failSafe) { return; } // TODO:
|
||||
if (lostFrame) { return; }
|
||||
void readRC() {
|
||||
if (RC.read()) {
|
||||
SBUSData data = RC.data();
|
||||
memcpy(channels, data.ch, sizeof(channels)); // copy channels data
|
||||
normalizeRC();
|
||||
}
|
||||
}
|
||||
|
||||
static void normalizeRC() {
|
||||
void normalizeRC() {
|
||||
for (uint8_t i = 0; i < RC_CHANNELS; i++) {
|
||||
controls[i] = mapf(channels[i], channelNeutral[i], channelMax[i], 0, 1);
|
||||
}
|
||||
}
|
||||
|
||||
void calibrateRC() {
|
||||
Serial.println("Calibrate RC: move all sticks to maximum positions within 4 seconds");
|
||||
Serial.println("··o ··o\n··· ···\n··· ···");
|
||||
delay(4000);
|
||||
for (int i = 0; i < 30; i++) readRC(); // ensure the values are updated
|
||||
for (int i = 0; i < RC_CHANNELS; i++) {
|
||||
channelMax[i] = channels[i];
|
||||
}
|
||||
Serial.println("Calibrate RC: move all sticks to neutral positions within 4 seconds");
|
||||
Serial.println("··· ···\n··· ·o·\n·o· ···");
|
||||
delay(4000);
|
||||
for (int i = 0; i < 30; i++) readRC(); // ensure the values are updated
|
||||
for (int i = 0; i < RC_CHANNELS; i++) {
|
||||
channelNeutral[i] = channels[i];
|
||||
}
|
||||
printRCCal();
|
||||
}
|
||||
|
||||
void printRCCal() {
|
||||
printArray(channelNeutral, RC_CHANNELS);
|
||||
printArray(channelMax, RC_CHANNELS);
|
||||
}
|
||||
|
||||
@@ -3,8 +3,7 @@
|
||||
|
||||
// Time related functions
|
||||
|
||||
void step()
|
||||
{
|
||||
void step() {
|
||||
float now = micros() / 1000000.0;
|
||||
dt = now - t;
|
||||
t = now;
|
||||
@@ -16,8 +15,7 @@ void step()
|
||||
computeLoopFreq();
|
||||
}
|
||||
|
||||
void computeLoopFreq()
|
||||
{
|
||||
void computeLoopFreq() {
|
||||
static float windowStart = 0;
|
||||
static uint32_t freq = 0;
|
||||
freq++;
|
||||
|
||||
@@ -3,31 +3,28 @@
|
||||
|
||||
// Utility functions
|
||||
|
||||
#include "math.h"
|
||||
#include <math.h>
|
||||
#include <soc/soc.h>
|
||||
#include <soc/rtc_cntl_reg.h>
|
||||
|
||||
float mapf(long x, long in_min, long in_max, float out_min, float out_max)
|
||||
{
|
||||
float mapf(long x, long in_min, long in_max, float out_min, float out_max) {
|
||||
return (float)(x - in_min) * (out_max - out_min) / (float)(in_max - in_min) + out_min;
|
||||
}
|
||||
|
||||
float mapff(float x, float in_min, float in_max, float out_min, float out_max)
|
||||
{
|
||||
float mapff(float x, float in_min, float in_max, float out_min, float out_max) {
|
||||
return (x - in_min) * (out_max - out_min) / (in_max - in_min) + out_min;
|
||||
}
|
||||
|
||||
int8_t sign(float x)
|
||||
{
|
||||
int8_t sign(float x) {
|
||||
return (x > 0) - (x < 0);
|
||||
}
|
||||
|
||||
float randomFloat(float min, float max)
|
||||
{
|
||||
float randomFloat(float min, float max) {
|
||||
return min + (max - min) * (float)rand() / RAND_MAX;
|
||||
}
|
||||
|
||||
// wrap angle to [-PI, PI)
|
||||
float wrapAngle(float angle)
|
||||
{
|
||||
// Wrap angle to [-PI, PI)
|
||||
float wrapAngle(float angle) {
|
||||
angle = fmodf(angle, 2 * PI);
|
||||
if (angle > PI) {
|
||||
angle -= 2 * PI;
|
||||
@@ -36,3 +33,18 @@ float wrapAngle(float angle)
|
||||
}
|
||||
return angle;
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
void printArray(T arr[], int size) {
|
||||
Serial.print("{");
|
||||
for (uint8_t i = 0; i < size; i++) {
|
||||
Serial.print(arr[i]);
|
||||
if (i < size - 1) Serial.print(", ");
|
||||
}
|
||||
Serial.println("}");
|
||||
}
|
||||
|
||||
// Disable reset on low voltage
|
||||
void disableBrownOut() {
|
||||
WRITE_PERI_REG(RTC_CNTL_BROWN_OUT_REG, 0);
|
||||
}
|
||||
|
||||
@@ -5,8 +5,7 @@
|
||||
|
||||
#pragma once
|
||||
|
||||
class Vector : public Printable
|
||||
{
|
||||
class Vector : public Printable {
|
||||
public:
|
||||
float x, y, z;
|
||||
|
||||
@@ -14,79 +13,79 @@ public:
|
||||
|
||||
Vector(float x, float y, float z): x(x), y(y), z(z) {};
|
||||
|
||||
float norm() const
|
||||
{
|
||||
float norm() const {
|
||||
return sqrt(x * x + y * y + z * z);
|
||||
}
|
||||
|
||||
bool zero() const
|
||||
{
|
||||
bool zero() const {
|
||||
return x == 0 && y == 0 && z == 0;
|
||||
}
|
||||
|
||||
void normalize()
|
||||
{
|
||||
void normalize() {
|
||||
float n = norm();
|
||||
x /= n;
|
||||
y /= n;
|
||||
z /= n;
|
||||
}
|
||||
|
||||
Vector operator * (const float b) const
|
||||
{
|
||||
Vector operator * (const float b) const {
|
||||
return Vector(x * b, y * b, z * b);
|
||||
}
|
||||
|
||||
Vector operator / (const float b) const
|
||||
{
|
||||
Vector operator / (const float b) const {
|
||||
return Vector(x / b, y / b, z / b);
|
||||
}
|
||||
|
||||
Vector operator + (const Vector& b) const
|
||||
{
|
||||
Vector operator + (const Vector& b) const {
|
||||
return Vector(x + b.x, y + b.y, z + b.z);
|
||||
}
|
||||
|
||||
Vector operator - (const Vector& b) const
|
||||
{
|
||||
Vector operator - (const Vector& b) const {
|
||||
return Vector(x - b.x, y - b.y, z - b.z);
|
||||
}
|
||||
|
||||
inline bool operator == (const Vector& b) const
|
||||
{
|
||||
// Element-wise multiplication
|
||||
Vector operator * (const Vector& b) const {
|
||||
return Vector(x * b.x, y * b.y, z * b.z);
|
||||
}
|
||||
|
||||
// Element-wise division
|
||||
Vector operator / (const Vector& b) const {
|
||||
return Vector(x / b.x, y / b.y, z / b.z);
|
||||
}
|
||||
|
||||
inline bool operator == (const Vector& b) const {
|
||||
return x == b.x && y == b.y && z == b.z;
|
||||
}
|
||||
|
||||
inline bool operator != (const Vector& b) const
|
||||
{
|
||||
inline bool operator != (const Vector& b) const {
|
||||
return !(*this == b);
|
||||
}
|
||||
|
||||
inline bool finite() const
|
||||
{
|
||||
inline bool finite() const {
|
||||
return isfinite(x) && isfinite(y) && isfinite(z);
|
||||
}
|
||||
|
||||
static float dot(const Vector& a, const Vector& b)
|
||||
{
|
||||
static float dot(const Vector& a, const Vector& b) {
|
||||
return a.x * b.x + a.y * b.y + a.z * b.z;
|
||||
}
|
||||
|
||||
static Vector cross(const Vector& a, const Vector& b)
|
||||
{
|
||||
static Vector cross(const Vector& a, const Vector& b) {
|
||||
return Vector(a.y * b.z - a.z * b.y, a.z * b.x - a.x * b.z, a.x * b.y - a.y * b.x);
|
||||
}
|
||||
|
||||
static float angleBetweenVectors(const Vector& a, const Vector& b)
|
||||
{
|
||||
static float angleBetweenVectors(const Vector& a, const Vector& b) {
|
||||
return acos(constrain(dot(a, b) / (a.norm() * b.norm()), -1, 1));
|
||||
}
|
||||
|
||||
static Vector angularRatesBetweenVectors(const Vector& u, const Vector& v)
|
||||
{
|
||||
Vector direction = cross(u, v);
|
||||
static Vector angularRatesBetweenVectors(const Vector& a, const Vector& b) {
|
||||
Vector direction = cross(a, b);
|
||||
if (direction.zero()) {
|
||||
// vectors are opposite, return any perpendicular vector
|
||||
return cross(a, Vector(1, 0, 0));
|
||||
}
|
||||
direction.normalize();
|
||||
float angle = angleBetweenVectors(u, v);
|
||||
float angle = angleBetweenVectors(a, b);
|
||||
return direction * angle;
|
||||
}
|
||||
|
||||
|
||||
@@ -6,10 +6,8 @@
|
||||
#if WIFI_ENABLED == 1
|
||||
|
||||
#include <WiFi.h>
|
||||
#include <WiFiClient.h>
|
||||
#include <WiFiAP.h>
|
||||
#include "SBUS.h"
|
||||
#include "mavlink/common/mavlink.h"
|
||||
#include <WiFiUdp.h>
|
||||
|
||||
#define WIFI_SSID "flix"
|
||||
#define WIFI_PASSWORD "flixwifi"
|
||||
@@ -18,18 +16,22 @@
|
||||
|
||||
WiFiUDP udp;
|
||||
|
||||
void setupWiFi()
|
||||
{
|
||||
void setupWiFi() {
|
||||
Serial.println("Setup Wi-Fi");
|
||||
WiFi.softAP(WIFI_SSID, WIFI_PASSWORD);
|
||||
IPAddress myIP = WiFi.softAPIP();
|
||||
udp.begin(WIFI_UDP_PORT);
|
||||
}
|
||||
|
||||
inline void sendWiFi(const uint8_t *buf, size_t len)
|
||||
{
|
||||
void sendWiFi(const uint8_t *buf, int len) {
|
||||
udp.beginPacket(WIFI_UDP_IP, WIFI_UDP_PORT);
|
||||
udp.write(buf, len);
|
||||
udp.endPacket();
|
||||
}
|
||||
|
||||
int receiveWiFi(uint8_t *buf, int len) {
|
||||
udp.parsePacket();
|
||||
return udp.read(buf, len);
|
||||
}
|
||||
|
||||
#endif
|
||||
|
||||
@@ -7,6 +7,7 @@
|
||||
|
||||
#include <cmath>
|
||||
#include <string>
|
||||
#include <stdint.h>
|
||||
#include <stdio.h>
|
||||
#include <unistd.h>
|
||||
#include <sys/poll.h>
|
||||
@@ -14,6 +15,8 @@
|
||||
#define PI 3.1415926535897932384626433832795
|
||||
#define DEG_TO_RAD 0.017453292519943295769236907684886
|
||||
#define RAD_TO_DEG 57.295779513082320876798154814105
|
||||
#define radians(deg) ((deg)*DEG_TO_RAD)
|
||||
#define degrees(rad) ((rad)*RAD_TO_DEG)
|
||||
|
||||
#define constrain(amt,low,high) ((amt)<(low)?(low):((amt)>(high)?(high):(amt)))
|
||||
|
||||
@@ -44,8 +47,7 @@ public:
|
||||
|
||||
class Print {
|
||||
public:
|
||||
size_t printf(const char *format, ...)
|
||||
{
|
||||
size_t printf(const char *format, ...) {
|
||||
va_list args;
|
||||
va_start(args, format);
|
||||
size_t result = vprintf(format, args);
|
||||
@@ -53,48 +55,43 @@ public:
|
||||
return result;
|
||||
}
|
||||
|
||||
size_t print(float n, int digits = 2)
|
||||
{
|
||||
size_t print(int n) {
|
||||
return printf("%d", n);
|
||||
}
|
||||
|
||||
size_t print(float n, int digits = 2) {
|
||||
return printf("%.*f", digits, n);
|
||||
}
|
||||
|
||||
size_t println(float n, int digits = 2)
|
||||
{
|
||||
size_t println(float n, int digits = 2) {
|
||||
return printf("%.*f\n", digits, n);
|
||||
}
|
||||
|
||||
size_t print(const char* s)
|
||||
{
|
||||
size_t print(const char* s) {
|
||||
return printf("%s", s);
|
||||
}
|
||||
|
||||
size_t println()
|
||||
{
|
||||
size_t println() {
|
||||
return print("\n");
|
||||
}
|
||||
|
||||
size_t println(const char* s)
|
||||
{
|
||||
size_t println(const char* s) {
|
||||
return printf("%s\n", s);
|
||||
}
|
||||
|
||||
size_t println(const Printable& p)
|
||||
{
|
||||
size_t println(const Printable& p) {
|
||||
return p.printTo(*this) + print("\n");
|
||||
}
|
||||
|
||||
size_t print(const String& s)
|
||||
{
|
||||
size_t print(const String& s) {
|
||||
return printf("%s", s.c_str());
|
||||
}
|
||||
|
||||
size_t println(const std::string& s)
|
||||
{
|
||||
size_t println(const std::string& s) {
|
||||
return printf("%s\n", s.c_str());
|
||||
}
|
||||
|
||||
size_t println(const String& s)
|
||||
{
|
||||
size_t println(const String& s) {
|
||||
return printf("%s\n", s.c_str());
|
||||
}
|
||||
};
|
||||
@@ -120,12 +117,14 @@ public:
|
||||
int read() {
|
||||
if (available()) {
|
||||
char c;
|
||||
::read(STDIN_FILENO, &c, 1); // use raw read to avoid C++ buffering
|
||||
size_t res = ::read(STDIN_FILENO, &c, 1); // use raw read to avoid C++ buffering
|
||||
// https://stackoverflow.com/questions/45238997/does-getchar-function-has-its-own-buffer-to-store-remaining-input
|
||||
return c;
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
void setRxInvert(bool invert) {};
|
||||
};
|
||||
|
||||
HardwareSerial Serial, Serial2;
|
||||
@@ -135,7 +134,8 @@ void delay(uint32_t ms) {
|
||||
}
|
||||
|
||||
unsigned long __micros;
|
||||
unsigned long __resetTime = 0;
|
||||
|
||||
unsigned long micros() {
|
||||
return __micros;
|
||||
return __micros + __resetTime; // keep the time monotonic
|
||||
}
|
||||
|
||||
@@ -1,4 +1,4 @@
|
||||
cmake_minimum_required(VERSION 3.0 FATAL_ERROR)
|
||||
cmake_minimum_required(VERSION 3.5 FATAL_ERROR)
|
||||
project(flix_gazebo)
|
||||
|
||||
# === gazebo plugin
|
||||
@@ -10,9 +10,13 @@ list(APPEND CMAKE_CXX_FLAGS "${GAZEBO_CXX_FLAGS}")
|
||||
|
||||
set(FLIX_SOURCE_DIR ../flix)
|
||||
include_directories(${FLIX_SOURCE_DIR})
|
||||
file(GLOB_RECURSE FLIX_INO_FILES ${FLIX_SOURCE_DIR}/*.ino)
|
||||
|
||||
set(CMAKE_BUILD_TYPE RelWithDebInfo)
|
||||
add_library(flix SHARED flix.cpp)
|
||||
add_library(flix SHARED simulator.cpp)
|
||||
target_link_libraries(flix ${GAZEBO_LIBRARIES} ${SDL2_LIBRARIES})
|
||||
target_include_directories(flix PUBLIC ${CMAKE_CURRENT_SOURCE_DIR})
|
||||
target_compile_options(flix PRIVATE -Wno-address-of-packed-member) # disable unneeded mavlink warnings
|
||||
|
||||
# Include dir for MAVLink-Arduino library
|
||||
target_include_directories(flix PUBLIC $ENV{HOME}/Arduino/libraries/MAVLink)
|
||||
target_include_directories(flix PUBLIC $ENV{HOME}/Documents/Arduino/libraries/MAVLink)
|
||||
|
||||
15
gazebo/README.md
Normal file
@@ -0,0 +1,15 @@
|
||||
# Gazebo Simulation
|
||||
|
||||
<img src="../docs/img/simulator.png" width=500 alt="Flix simulator">
|
||||
|
||||
## Building and running
|
||||
|
||||
See [building and running instructions](../docs/build.md#simulation).
|
||||
|
||||
## Code structure
|
||||
|
||||
Flix simulator is based on [Gazebo Classic](https://classic.gazebosim.org) and consists of the following components:
|
||||
|
||||
* Physical model of the drone: [`models/flix/flix.sdf`](models/flix/flix.sdf).
|
||||
* Plugin for Gazebo: [`simulator.cpp`](simulator.cpp). The plugin is attached to the physical model. It receives stick positions from the controller, gets the data from the virtual sensors, and then passes this data to the Arduino code.
|
||||
* Arduino imitation: [`Arduino.h`](Arduino.h). This file contains partial implementation of the Arduino API, that is working within Gazebo plugin environment.
|
||||
25
gazebo/SBUS.h
Normal file
@@ -0,0 +1,25 @@
|
||||
// Copyright (c) 2023 Oleg Kalachev <okalachev@gmail.com>
|
||||
// Repository: https://github.com/okalachev/flix
|
||||
|
||||
// SBUS library mock to make it possible to compile simulator with rc.ino
|
||||
|
||||
#include "joystick.h"
|
||||
|
||||
struct SBUSData {
|
||||
int16_t ch[16];
|
||||
};
|
||||
|
||||
class SBUS {
|
||||
public:
|
||||
SBUS(HardwareSerial& bus, const bool inv = true) {};
|
||||
SBUS(HardwareSerial& bus, const int8_t rxpin, const int8_t txpin, const bool inv = true) {};
|
||||
void begin() {};
|
||||
bool read() { return joystickGet(); };
|
||||
SBUSData data() {
|
||||
SBUSData data;
|
||||
for (uint8_t i = 0; i < 16; i++) {
|
||||
data.ch[i] = channels[i];
|
||||
}
|
||||
return data;
|
||||
};
|
||||
};
|
||||
209
gazebo/flix.cpp
@@ -1,209 +0,0 @@
|
||||
// Copyright (c) 2023 Oleg Kalachev <okalachev@gmail.com>
|
||||
// Repository: https://github.com/okalachev/flix
|
||||
|
||||
// Gazebo plugin for running Arduino code and simulating the drone
|
||||
|
||||
#include <functional>
|
||||
#include <cmath>
|
||||
#include <gazebo/gazebo.hh>
|
||||
#include <gazebo/physics/physics.hh>
|
||||
#include <gazebo/rendering/rendering.hh>
|
||||
#include <gazebo/common/common.hh>
|
||||
#include <gazebo/sensors/sensors.hh>
|
||||
#include <gazebo/msgs/msgs.hh>
|
||||
#include <ignition/math/Vector3.hh>
|
||||
#include <ignition/math/Pose3.hh>
|
||||
#include <ignition/math/Quaternion.hh>
|
||||
#include <iostream>
|
||||
#include <fstream>
|
||||
|
||||
#include "Arduino.h"
|
||||
#include "flix.h"
|
||||
#include "util.ino"
|
||||
#include "joystick.h"
|
||||
#include "time.ino"
|
||||
#include "estimate.ino"
|
||||
#include "control.ino"
|
||||
#include "log.ino"
|
||||
#include "cli.ino"
|
||||
#include "lpf.h"
|
||||
|
||||
using ignition::math::Vector3d;
|
||||
using ignition::math::Pose3d;
|
||||
using namespace gazebo;
|
||||
using namespace std;
|
||||
|
||||
Pose3d flu2frd(const Pose3d& p)
|
||||
{
|
||||
return ignition::math::Pose3d(p.Pos().X(), -p.Pos().Y(), -p.Pos().Z(),
|
||||
p.Rot().W(), p.Rot().X(), -p.Rot().Y(), -p.Rot().Z());
|
||||
}
|
||||
|
||||
Vector flu2frd(const Vector3d& v)
|
||||
{
|
||||
return Vector(v.X(), -v.Y(), -v.Z());
|
||||
}
|
||||
|
||||
class ModelFlix : public ModelPlugin
|
||||
{
|
||||
private:
|
||||
physics::ModelPtr model, estimateModel;
|
||||
physics::LinkPtr body;
|
||||
sensors::ImuSensorPtr imu;
|
||||
event::ConnectionPtr updateConnection, resetConnection;
|
||||
transport::NodePtr nodeHandle;
|
||||
transport::PublisherPtr motorPub[4];
|
||||
LowPassFilter<Vector> accFilter = LowPassFilter<Vector>(0.1);
|
||||
|
||||
public:
|
||||
void Load(physics::ModelPtr _parent, sdf::ElementPtr /*_sdf*/)
|
||||
{
|
||||
this->model = _parent;
|
||||
this->body = this->model->GetLink("body");
|
||||
|
||||
this->imu = std::dynamic_pointer_cast<sensors::ImuSensor>(sensors::get_sensor(model->GetScopedName(true) + "::body::imu")); // default::flix::body::imu
|
||||
if (imu == nullptr) {
|
||||
gzerr << "IMU sensor not found" << std::endl;
|
||||
return;
|
||||
}
|
||||
|
||||
this->estimateModel = model->GetWorld()->ModelByName("flix_estimate");
|
||||
|
||||
this->updateConnection = event::Events::ConnectWorldUpdateBegin(
|
||||
std::bind(&ModelFlix::OnUpdate, this));
|
||||
|
||||
this->resetConnection = event::Events::ConnectWorldReset(
|
||||
std::bind(&ModelFlix::OnReset, this));
|
||||
|
||||
initNode();
|
||||
|
||||
Serial.begin(0);
|
||||
|
||||
gzmsg << "Flix plugin loaded" << endl;
|
||||
}
|
||||
|
||||
public:
|
||||
void OnReset()
|
||||
{
|
||||
attitude = Quaternion();
|
||||
gzmsg << "Flix plugin reset" << endl;
|
||||
}
|
||||
|
||||
void OnUpdate()
|
||||
{
|
||||
__micros = model->GetWorld()->SimTime().Double() * 1000000;
|
||||
step();
|
||||
|
||||
// read imu
|
||||
rates = flu2frd(imu->AngularVelocity());
|
||||
acc = this->accFilter.update(flu2frd(imu->LinearAcceleration()));
|
||||
|
||||
// read rc
|
||||
joystickGet();
|
||||
controls[RC_CHANNEL_MODE] = 1; // 0 acro, 1 stab
|
||||
controls[RC_CHANNEL_AUX] = 1; // armed
|
||||
|
||||
estimate();
|
||||
|
||||
// correct yaw to the actual yaw
|
||||
attitude.setYaw(-this->model->WorldPose().Yaw());
|
||||
|
||||
control();
|
||||
parseInput();
|
||||
|
||||
applyMotorsThrust();
|
||||
updateEstimatePose();
|
||||
publishTopics();
|
||||
logData();
|
||||
}
|
||||
|
||||
void applyMotorsThrust()
|
||||
{
|
||||
// thrusts
|
||||
const double d = 0.035355;
|
||||
const double maxThrust = 0.03 * ONE_G; // 30 g, https://www.youtube.com/watch?v=VtKI4Pjx8Sk
|
||||
// 65 mm prop ~40 g
|
||||
|
||||
const float scale0 = 1.0, scale1 = 1.1, scale2 = 0.9, scale3 = 1.05;
|
||||
const float minThrustRel = 0;
|
||||
|
||||
// apply min thrust
|
||||
float mfl = mapff(motors[MOTOR_FRONT_LEFT], 0, 1, minThrustRel, 1);
|
||||
float mfr = mapff(motors[MOTOR_FRONT_RIGHT], 0, 1, minThrustRel, 1);
|
||||
float mrl = mapff(motors[MOTOR_REAR_LEFT], 0, 1, minThrustRel, 1);
|
||||
float mrr = mapff(motors[MOTOR_REAR_RIGHT], 0, 1, minThrustRel, 1);
|
||||
|
||||
if (motors[MOTOR_FRONT_LEFT] < 0.001) mfl = 0;
|
||||
if (motors[MOTOR_FRONT_RIGHT] < 0.001) mfr = 0;
|
||||
if (motors[MOTOR_REAR_LEFT] < 0.001) mrl = 0;
|
||||
if (motors[MOTOR_REAR_RIGHT] < 0.001) mrr = 0;
|
||||
|
||||
// TODO: min_thrust
|
||||
|
||||
body->AddLinkForce(Vector3d(0.0, 0.0, scale0 * maxThrust * abs(mfl)), Vector3d(d, d, 0.0));
|
||||
body->AddLinkForce(Vector3d(0.0, 0.0, scale1 * maxThrust * abs(mfr)), Vector3d(d, -d, 0.0));
|
||||
body->AddLinkForce(Vector3d(0.0, 0.0, scale2 * maxThrust * abs(mrl)), Vector3d(-d, d, 0.0));
|
||||
body->AddLinkForce(Vector3d(0.0, 0.0, scale3 * maxThrust * abs(mrr)), Vector3d(-d, -d, 0.0));
|
||||
|
||||
// TODO: indicate if > 1
|
||||
|
||||
// torque
|
||||
const double maxTorque = 0.0023614413; // 24.08 g*cm
|
||||
int direction = 1;
|
||||
// z is counter clockwise, normal rotation direction is minus
|
||||
body->AddRelativeTorque(Vector3d(0.0, 0.0, direction * scale0 * maxTorque * motors[MOTOR_FRONT_LEFT]));
|
||||
body->AddRelativeTorque(Vector3d(0.0, 0.0, direction * scale1 * -maxTorque * motors[MOTOR_FRONT_RIGHT]));
|
||||
body->AddRelativeTorque(Vector3d(0.0, 0.0, direction * scale2 * -maxTorque * motors[MOTOR_REAR_LEFT]));
|
||||
body->AddRelativeTorque(Vector3d(0.0, 0.0, direction * scale3 * maxTorque * motors[MOTOR_REAR_RIGHT]));
|
||||
}
|
||||
|
||||
void updateEstimatePose() {
|
||||
if (estimateModel == nullptr) {
|
||||
return;
|
||||
}
|
||||
if (!attitude.finite()) {
|
||||
// gzerr << "attitude is nan" << std::endl;
|
||||
return;
|
||||
}
|
||||
Pose3d pose(
|
||||
model->WorldPose().Pos().X(), model->WorldPose().Pos().Y(), model->WorldPose().Pos().Z(),
|
||||
attitude.w, attitude.x, -attitude.y, -attitude.z // frd to flu
|
||||
);
|
||||
// std::cout << pose.Pos().X() << " " << pose.Pos().Y() << " " << pose.Pos().Z() <<
|
||||
// " " << pose.Rot().W() << " " << pose.Rot().X() << " " << pose.Rot().Y() << " " << pose.Rot().Z() << std::endl;
|
||||
|
||||
// calculate attitude estimation error
|
||||
|
||||
Quaternion groundtruthAttitude(estimateModel->WorldPose().Rot().W(), estimateModel->WorldPose().Rot().X(), -estimateModel->WorldPose().Rot().Y(), -estimateModel->WorldPose().Rot().Z());
|
||||
float angle = Vector::angleBetweenVectors(attitude.rotate(Vector(0, 0, -1)), groundtruthAttitude.rotate(Vector(0, 0, -1)));
|
||||
if (angle < 0.3) {
|
||||
//gzwarn << "att err: " << angle << endl;
|
||||
// TODO: warning
|
||||
// position under the floor to make it invisible
|
||||
pose.SetZ(-5);
|
||||
}
|
||||
|
||||
estimateModel->SetWorldPose(pose);
|
||||
|
||||
}
|
||||
|
||||
void initNode() {
|
||||
nodeHandle = transport::NodePtr(new transport::Node());
|
||||
nodeHandle->Init();
|
||||
string ns = "~/" + model->GetName();
|
||||
motorPub[0] = nodeHandle->Advertise<msgs::Int>(ns + "/motor0");
|
||||
motorPub[1] = nodeHandle->Advertise<msgs::Int>(ns + "/motor1");
|
||||
motorPub[2] = nodeHandle->Advertise<msgs::Int>(ns + "/motor2");
|
||||
motorPub[3] = nodeHandle->Advertise<msgs::Int>(ns + "/motor3");
|
||||
}
|
||||
|
||||
void publishTopics() {
|
||||
for (int i = 0; i < 4; i++) {
|
||||
msgs::Int msg;
|
||||
msg.set_data(static_cast<int>(std::round(motors[i] * 1000)));
|
||||
motorPub[i]->Publish(msg);
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
GZ_REGISTER_MODEL_PLUGIN(ModelFlix)
|
||||
@@ -8,21 +8,25 @@
|
||||
#include "vector.h"
|
||||
#include "quaternion.h"
|
||||
#include "Arduino.h"
|
||||
#include "wifi.h"
|
||||
|
||||
#define RC_CHANNELS 6
|
||||
#define RC_CHANNELS 16
|
||||
|
||||
#define MOTOR_REAR_LEFT 0
|
||||
#define MOTOR_FRONT_LEFT 3
|
||||
#define MOTOR_FRONT_RIGHT 2
|
||||
#define MOTOR_REAR_RIGHT 1
|
||||
|
||||
#define WIFI_ENABLED 1
|
||||
|
||||
float t = NAN;
|
||||
float dt;
|
||||
float loopFreq;
|
||||
float motors[4];
|
||||
int16_t channels[16]; // raw rc channels WARNING: unsigned on hardware
|
||||
int16_t channels[16]; // raw rc channels
|
||||
float controls[RC_CHANNELS];
|
||||
Vector acc;
|
||||
Vector gyro;
|
||||
Vector rates;
|
||||
Quaternion attitude;
|
||||
|
||||
@@ -39,11 +43,18 @@ void controlTorque();
|
||||
void showTable();
|
||||
bool motorsActive();
|
||||
void cliTestMotor(uint8_t n);
|
||||
void printRCCal();
|
||||
void processMavlink();
|
||||
void sendMavlink();
|
||||
void sendMessage(const void *msg);
|
||||
void receiveMavlink();
|
||||
void handleMavlink(const void *_msg);
|
||||
inline Quaternion FLU2FRD(const Quaternion &q);
|
||||
|
||||
// mocks
|
||||
void setLED(bool on) {};
|
||||
void calibrateGyro() { printf("Skip gyro calibrating\n"); };
|
||||
void calibrateAccel() { printf("Skip accel calibrating\n"); };
|
||||
void fullMotorTest(int n, bool reverse) { printf("Skip full motor test\n"); };
|
||||
void fullMotorTest(int n) { printf("Skip full motor test\n"); };
|
||||
void sendMotors() {};
|
||||
void printIMUCal() { printf("cal: N/A\n"); };
|
||||
|
||||
@@ -9,7 +9,7 @@
|
||||
</scene>
|
||||
<gui>
|
||||
<camera name="user_camera">
|
||||
<pose>-2 -0.3 1.5 0 0.5 0.1</pose>
|
||||
<pose>-2.3 0 1.1 0 0.3 0</pose>
|
||||
</camera>
|
||||
</gui>
|
||||
<physics type="ode">
|
||||
@@ -23,20 +23,7 @@
|
||||
</include>
|
||||
<include>
|
||||
<uri>model://flix</uri>
|
||||
<pose>0 0 0.2 0 0 0</pose>
|
||||
<pose>0 0 0.3 0 0 0</pose>
|
||||
</include>
|
||||
<model name="flix_estimate">
|
||||
<static>true</static>
|
||||
<link name="estimate">
|
||||
<visual name="estimate">
|
||||
<pose>0 0 0 0 0 1.57</pose>
|
||||
<geometry>
|
||||
<box>
|
||||
<size>0.125711 0.125711 0.022</size>
|
||||
</box>
|
||||
</geometry>
|
||||
</visual>
|
||||
</link>
|
||||
</model>
|
||||
</world>
|
||||
</sdf>
|
||||
|
||||
@@ -7,64 +7,50 @@
|
||||
#include <gazebo/gazebo.hh>
|
||||
#include <iostream>
|
||||
|
||||
using namespace std;
|
||||
|
||||
static const int16_t channelNeutralMin[] = {-1290, -258, -26833, 0, 0, 0};
|
||||
static const int16_t channelNeutralMax[] = {-1032, -258, -27348, 3353, 0, 0};
|
||||
|
||||
static const int16_t channelMax[] = {27090, 27090, 27090, 27090, 0, 0};
|
||||
// simulation calibration overrides, NOTE: use `cr` command and replace with the actual values
|
||||
const int channelNeutralOverride[] = {-258, -258, -27349, 0, -27349, 0};
|
||||
const int channelMaxOverride[] = {27090, 27090, 27090, 27090, -5676, 1};
|
||||
|
||||
#define RC_CHANNEL_ROLL 0
|
||||
#define RC_CHANNEL_PITCH 1
|
||||
#define RC_CHANNEL_THROTTLE 2
|
||||
#define RC_CHANNEL_YAW 3
|
||||
#define RC_CHANNEL_AUX 4
|
||||
#define RC_CHANNEL_MODE 5
|
||||
|
||||
static SDL_Joystick *joystick;
|
||||
#define RC_CHANNEL_ARMED 5
|
||||
#define RC_CHANNEL_MODE 4
|
||||
|
||||
SDL_Joystick *joystick;
|
||||
bool joystickInitialized = false, warnShown = false;
|
||||
|
||||
void normalizeRC();
|
||||
|
||||
void joystickInit()
|
||||
{
|
||||
void joystickInit() {
|
||||
SDL_Init(SDL_INIT_JOYSTICK);
|
||||
joystick = SDL_JoystickOpen(0);
|
||||
if (joystick != NULL) {
|
||||
joystickInitialized = true;
|
||||
gzmsg << "Joystick initialized: " << SDL_JoystickNameForIndex(0) << endl;
|
||||
gzmsg << "Joystick initialized: " << SDL_JoystickNameForIndex(0) << std::endl;
|
||||
} else if (!warnShown) {
|
||||
gzwarn << "Joystick not found, begin waiting for joystick..." << endl;
|
||||
gzwarn << "Joystick not found, begin waiting for joystick..." << std::endl;
|
||||
warnShown = true;
|
||||
}
|
||||
|
||||
// apply calibration overrides
|
||||
extern int channelNeutral[RC_CHANNELS];
|
||||
extern int channelMax[RC_CHANNELS];
|
||||
memcpy(channelNeutral, channelNeutralOverride, sizeof(channelNeutralOverride));
|
||||
memcpy(channelMax, channelMaxOverride, sizeof(channelMaxOverride));
|
||||
}
|
||||
|
||||
void joystickGet()
|
||||
{
|
||||
bool joystickGet() {
|
||||
if (!joystickInitialized) {
|
||||
joystickInit();
|
||||
return;
|
||||
return false;
|
||||
}
|
||||
|
||||
SDL_JoystickUpdate();
|
||||
|
||||
for (uint8_t i = 0; i < 4; i++) {
|
||||
for (uint8_t i = 0; i < 8; i++) {
|
||||
channels[i] = SDL_JoystickGetAxis(joystick, i);
|
||||
}
|
||||
channels[RC_CHANNEL_MODE] = SDL_JoystickGetButton(joystick, 0) ? 1 : 0;
|
||||
controls[RC_CHANNEL_MODE] = channels[RC_CHANNEL_MODE];
|
||||
|
||||
normalizeRC();
|
||||
}
|
||||
|
||||
void normalizeRC() {
|
||||
for (uint8_t i = 0; i < 4; i++) {
|
||||
if (channels[i] >= channelNeutralMin[i] && channels[i] <= channelNeutralMax[i]) {
|
||||
controls[i] = 0;
|
||||
} else {
|
||||
controls[i] = mapf(channels[i], (channelNeutralMin[i] + channelNeutralMax[i]) / 2, channelMax[i], 0, 1);
|
||||
}
|
||||
}
|
||||
controls[RC_CHANNEL_THROTTLE] = constrain(controls[RC_CHANNEL_THROTTLE], 0, 1);
|
||||
return true;
|
||||
}
|
||||
|
||||
@@ -13,15 +13,10 @@
|
||||
<collision name="collision">
|
||||
<geometry>
|
||||
<box>
|
||||
<size>0.125711 0.125711 0.022</size>
|
||||
<size>0.095 0.095 0.0276</size>
|
||||
</box>
|
||||
</geometry>
|
||||
</collision>
|
||||
<visual name="body">
|
||||
<geometry>
|
||||
<mesh><uri>model://flix/flix.dae</uri></mesh>
|
||||
</geometry>
|
||||
</visual>
|
||||
<sensor name="imu" type="imu">
|
||||
<always_on>1</always_on>
|
||||
<visualize>1</visualize>
|
||||
@@ -63,6 +58,37 @@
|
||||
</linear_acceleration>
|
||||
</imu>
|
||||
</sensor>
|
||||
<visual name="body">
|
||||
<geometry>
|
||||
<mesh><uri>model://flix/flix.stl</uri></mesh>
|
||||
</geometry>
|
||||
<material>
|
||||
<ambient>0.5 0.5 0.6 1</ambient>
|
||||
<diffuse>0.5 0.5 0.6 1</diffuse>
|
||||
<specular>0 0 0 1</specular>
|
||||
<emissive>0 0 0 1</emissive>
|
||||
</material>
|
||||
</visual>
|
||||
<visual name="prop0"><!-- rear left -->
|
||||
<geometry><cylinder><radius>0.0275</radius><length>0</length></cylinder></geometry>
|
||||
<pose>-0.04243 0.04243 0.0142 0 0 0</pose>
|
||||
<material><ambient>0.8 0.3 0.3 0.5</ambient><diffuse>0.8 0.3 0.3 0.5</diffuse></material>
|
||||
</visual>
|
||||
<visual name="prop1"><!-- rear right -->
|
||||
<geometry><cylinder><radius>0.0275</radius><length>0</length></cylinder></geometry>
|
||||
<pose>-0.04243 -0.04243 0.0142 0 0 0</pose>
|
||||
<material><ambient>0.8 0.3 0.3 0.5</ambient><diffuse>0.8 0.3 0.3 0.5</diffuse></material>
|
||||
</visual>
|
||||
<visual name="prop2"><!-- front right -->
|
||||
<geometry><cylinder><radius>0.0275</radius><length>0</length></cylinder></geometry>
|
||||
<pose>0.04243 -0.04243 0.0142 0 0 0</pose>
|
||||
<material><ambient>1 1 1 0.5</ambient><diffuse>1 1 1 0.5</diffuse></material>
|
||||
</visual>
|
||||
<visual name="prop3"><!-- front left -->
|
||||
<geometry><cylinder><radius>0.0275</radius><length>0</length></cylinder></geometry>
|
||||
<pose>0.04243 0.04243 0.0142 0 0 0</pose>
|
||||
<material><ambient>1 1 1 0.5</ambient><diffuse>1 1 1 0.5</diffuse></material>
|
||||
</visual>
|
||||
</link>
|
||||
<plugin name="flix" filename="libflix.so"/>
|
||||
</model>
|
||||
|
||||
BIN
gazebo/models/flix/flix.stl
Normal file
133
gazebo/simulator.cpp
Normal file
@@ -0,0 +1,133 @@
|
||||
// Copyright (c) 2023 Oleg Kalachev <okalachev@gmail.com>
|
||||
// Repository: https://github.com/okalachev/flix
|
||||
|
||||
// Gazebo plugin for running Arduino code and simulating the drone
|
||||
|
||||
#include <functional>
|
||||
#include <cmath>
|
||||
#include <gazebo/gazebo.hh>
|
||||
#include <gazebo/physics/physics.hh>
|
||||
#include <gazebo/common/common.hh>
|
||||
#include <gazebo/sensors/sensors.hh>
|
||||
#include <gazebo/msgs/msgs.hh>
|
||||
#include <ignition/math/Vector3.hh>
|
||||
#include <ignition/math/Pose3.hh>
|
||||
#include <iostream>
|
||||
#include <fstream>
|
||||
|
||||
#include "Arduino.h"
|
||||
#include "flix.h"
|
||||
#include "util.ino"
|
||||
#include "rc.ino"
|
||||
#include "time.ino"
|
||||
#include "estimate.ino"
|
||||
#include "control.ino"
|
||||
#include "log.ino"
|
||||
#include "cli.ino"
|
||||
#include "mavlink.ino"
|
||||
#include "lpf.h"
|
||||
|
||||
using ignition::math::Vector3d;
|
||||
using namespace gazebo;
|
||||
using namespace std;
|
||||
|
||||
class ModelFlix : public ModelPlugin {
|
||||
private:
|
||||
physics::ModelPtr model;
|
||||
physics::LinkPtr body;
|
||||
sensors::ImuSensorPtr imu;
|
||||
event::ConnectionPtr updateConnection, resetConnection;
|
||||
transport::NodePtr nodeHandle;
|
||||
transport::PublisherPtr motorPub[4];
|
||||
LowPassFilter<Vector> accFilter = LowPassFilter<Vector>(0.1);
|
||||
|
||||
public:
|
||||
void Load(physics::ModelPtr _parent, sdf::ElementPtr /*_sdf*/) {
|
||||
this->model = _parent;
|
||||
this->body = this->model->GetLink("body");
|
||||
this->imu = dynamic_pointer_cast<sensors::ImuSensor>(sensors::get_sensor(model->GetScopedName(true) + "::body::imu")); // default::flix::body::imu
|
||||
this->updateConnection = event::Events::ConnectWorldUpdateBegin(std::bind(&ModelFlix::OnUpdate, this));
|
||||
this->resetConnection = event::Events::ConnectWorldReset(std::bind(&ModelFlix::OnReset, this));
|
||||
initNode();
|
||||
Serial.begin(0);
|
||||
gzmsg << "Flix plugin loaded" << endl;
|
||||
}
|
||||
|
||||
void OnReset() {
|
||||
attitude = Quaternion(); // reset estimated attitude
|
||||
__resetTime += __micros;
|
||||
gzmsg << "Flix plugin reset" << endl;
|
||||
}
|
||||
|
||||
void OnUpdate() {
|
||||
__micros = model->GetWorld()->SimTime().Double() * 1000000;
|
||||
step();
|
||||
|
||||
// read virtual imu
|
||||
gyro = Vector(imu->AngularVelocity().X(), imu->AngularVelocity().Y(), imu->AngularVelocity().Z());
|
||||
acc = this->accFilter.update(Vector(imu->LinearAcceleration().X(), imu->LinearAcceleration().Y(), imu->LinearAcceleration().Z()));
|
||||
|
||||
// read rc
|
||||
readRC();
|
||||
controls[RC_CHANNEL_MODE] = 1; // 0 acro, 1 stab
|
||||
controls[RC_CHANNEL_ARMED] = 1; // armed
|
||||
|
||||
estimate();
|
||||
|
||||
// correct yaw to the actual yaw
|
||||
attitude.setYaw(this->model->WorldPose().Yaw());
|
||||
|
||||
control();
|
||||
parseInput();
|
||||
processMavlink();
|
||||
|
||||
applyMotorForces();
|
||||
publishTopics();
|
||||
logData();
|
||||
}
|
||||
|
||||
void applyMotorForces() {
|
||||
// thrusts
|
||||
const double dist = 0.035355; // motors shift from the center, m
|
||||
const double maxThrust = 0.03 * ONE_G; // ~30 g, https://youtu.be/VtKI4Pjx8Sk?&t=78
|
||||
|
||||
const float scale0 = 1.0, scale1 = 1.1, scale2 = 0.9, scale3 = 1.05; // imitating motors asymmetry
|
||||
float mfl = scale0 * maxThrust * motors[MOTOR_FRONT_LEFT];
|
||||
float mfr = scale1 * maxThrust * motors[MOTOR_FRONT_RIGHT];
|
||||
float mrl = scale2 * maxThrust * motors[MOTOR_REAR_LEFT];
|
||||
float mrr = scale3 * maxThrust * motors[MOTOR_REAR_RIGHT];
|
||||
|
||||
body->AddLinkForce(Vector3d(0.0, 0.0, mfl), Vector3d(dist, dist, 0.0));
|
||||
body->AddLinkForce(Vector3d(0.0, 0.0, mfr), Vector3d(dist, -dist, 0.0));
|
||||
body->AddLinkForce(Vector3d(0.0, 0.0, mrl), Vector3d(-dist, dist, 0.0));
|
||||
body->AddLinkForce(Vector3d(0.0, 0.0, mrr), Vector3d(-dist, -dist, 0.0));
|
||||
|
||||
// torque
|
||||
const double maxTorque = 0.0024 * ONE_G; // ~24 g*cm
|
||||
body->AddRelativeTorque(Vector3d(0.0, 0.0, scale0 * maxTorque * motors[MOTOR_FRONT_LEFT]));
|
||||
body->AddRelativeTorque(Vector3d(0.0, 0.0, scale1 * -maxTorque * motors[MOTOR_FRONT_RIGHT]));
|
||||
body->AddRelativeTorque(Vector3d(0.0, 0.0, scale2 * -maxTorque * motors[MOTOR_REAR_LEFT]));
|
||||
body->AddRelativeTorque(Vector3d(0.0, 0.0, scale3 * maxTorque * motors[MOTOR_REAR_RIGHT]));
|
||||
}
|
||||
|
||||
void initNode() {
|
||||
nodeHandle = transport::NodePtr(new transport::Node());
|
||||
nodeHandle->Init();
|
||||
string ns = "~/" + model->GetName();
|
||||
// create motors output topics for debugging and plotting
|
||||
motorPub[0] = nodeHandle->Advertise<msgs::Int>(ns + "/motor0");
|
||||
motorPub[1] = nodeHandle->Advertise<msgs::Int>(ns + "/motor1");
|
||||
motorPub[2] = nodeHandle->Advertise<msgs::Int>(ns + "/motor2");
|
||||
motorPub[3] = nodeHandle->Advertise<msgs::Int>(ns + "/motor3");
|
||||
}
|
||||
|
||||
void publishTopics() {
|
||||
for (int i = 0; i < 4; i++) {
|
||||
msgs::Int msg;
|
||||
msg.set_data(static_cast<int>(round(motors[i] * 1000)));
|
||||
motorPub[i]->Publish(msg);
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
GZ_REGISTER_MODEL_PLUGIN(ModelFlix)
|
||||
1
gazebo/soc/rtc_cntl_reg.h
Normal file
@@ -0,0 +1 @@
|
||||
// Dummy file to make it possible to compile simulator with util.ino
|
||||
3
gazebo/soc/soc.h
Normal file
@@ -0,0 +1,3 @@
|
||||
// Dummy file to make it possible to compile simulator with util.ino
|
||||
|
||||
#define WRITE_PERI_REG(addr, val) {}
|
||||
44
gazebo/wifi.h
Normal file
@@ -0,0 +1,44 @@
|
||||
// Copyright (c) 2023 Oleg Kalachev <okalachev@gmail.com>
|
||||
// Repository: https://github.com/okalachev/flix
|
||||
|
||||
// sendWiFi and receiveWiFi implementations for the simulation
|
||||
|
||||
#include <arpa/inet.h>
|
||||
#include <netinet/in.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/socket.h>
|
||||
#include <unistd.h>
|
||||
#include <sys/poll.h>
|
||||
#include <gazebo/gazebo.hh>
|
||||
|
||||
#define WIFI_UDP_PORT_LOCAL 14580
|
||||
#define WIFI_UDP_PORT_REMOTE 14550
|
||||
|
||||
int wifiSocket;
|
||||
|
||||
void setupWiFi() {
|
||||
wifiSocket = socket(AF_INET, SOCK_DGRAM, 0);
|
||||
sockaddr_in addr; // local address
|
||||
addr.sin_family = AF_INET;
|
||||
addr.sin_addr.s_addr = INADDR_ANY;
|
||||
addr.sin_port = htons(WIFI_UDP_PORT_LOCAL);
|
||||
bind(wifiSocket, (sockaddr *)&addr, sizeof(addr));
|
||||
int broadcast = 1;
|
||||
setsockopt(wifiSocket, SOL_SOCKET, SO_BROADCAST, &broadcast, sizeof(broadcast)); // enable broadcast
|
||||
gzmsg << "WiFi UDP socket initialized on port " << WIFI_UDP_PORT_LOCAL << " (remote port " << WIFI_UDP_PORT_REMOTE << ")" << std::endl;
|
||||
}
|
||||
|
||||
void sendWiFi(const uint8_t *buf, int len) {
|
||||
if (wifiSocket == 0) setupWiFi();
|
||||
sockaddr_in addr; // remote address
|
||||
addr.sin_family = AF_INET;
|
||||
addr.sin_addr.s_addr = INADDR_BROADCAST; // send UDP broadcast
|
||||
addr.sin_port = htons(WIFI_UDP_PORT_REMOTE);
|
||||
sendto(wifiSocket, buf, len, 0, (sockaddr *)&addr, sizeof(addr));
|
||||
}
|
||||
|
||||
int receiveWiFi(uint8_t *buf, int len) {
|
||||
struct pollfd pfd = { .fd = wifiSocket, .events = POLLIN };
|
||||
if (poll(&pfd, 1, 0) <= 0) return 0; // check if there is data to read
|
||||
return recv(wifiSocket, buf, len, 0);
|
||||
}
|
||||
61
tools/check_c_cpp_properties.py
Executable file
@@ -0,0 +1,61 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
import os
|
||||
import platform
|
||||
import json
|
||||
import re
|
||||
|
||||
path = '.vscode/c_cpp_properties.json' if os.path.exists('./.vscode/c_cpp_properties.json') else '../.vscode/c_cpp_properties.json'
|
||||
txt = open(path).read()
|
||||
# remove comments
|
||||
txt = re.sub(r'//.*', '', txt)
|
||||
props = json.loads(txt)
|
||||
|
||||
env = props.get('env', {})
|
||||
env['workspaceFolder'] = '.'
|
||||
|
||||
def check_path(s):
|
||||
source = s
|
||||
# replace env
|
||||
for key, value in env.items():
|
||||
s = s.replace('${' + key + '}', value)
|
||||
# remove globs from the end
|
||||
if s.endswith('**'):
|
||||
s = s[:-2]
|
||||
elif s.endswith('*'):
|
||||
s = s[:-1]
|
||||
s = os.path.expanduser(s)
|
||||
if s == '':
|
||||
s = '.'
|
||||
print('Check', source, '->', s)
|
||||
assert os.path.exists(s), 'Path does not exist: ' + s
|
||||
|
||||
# linux, macos or windows:
|
||||
platform = platform.system().lower()
|
||||
if platform == 'darwin':
|
||||
platform = 'mac'
|
||||
elif platform == 'windows':
|
||||
platform = 'win32'
|
||||
elif platform == 'linux':
|
||||
pass
|
||||
else:
|
||||
raise Exception('Unknown platform: ' + platform)
|
||||
|
||||
for configuration in props['configurations']:
|
||||
if platform not in configuration['name'].lower():
|
||||
print('Skip configuration', configuration['name'])
|
||||
continue
|
||||
|
||||
print('Check configuration', configuration['name'])
|
||||
|
||||
for include_path in configuration.get('includePath', []):
|
||||
check_path(include_path)
|
||||
|
||||
for forced_include in configuration.get('forcedInclude', []):
|
||||
check_path(forced_include)
|
||||
|
||||
for browse in configuration.get('browse', {}).get('path', []):
|
||||
check_path(browse)
|
||||
|
||||
if 'compilerPath' in configuration:
|
||||
check_path(configuration['compilerPath'])
|
||||
46
tools/csv_to_mcap.py
Executable file
@@ -0,0 +1,46 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
"""Convert CSV log file to MCAP file.
|
||||
|
||||
Usage:
|
||||
csv_to_mcap.py <csv_file> [<mcap_file>]
|
||||
"""
|
||||
|
||||
import csv
|
||||
import json
|
||||
import docopt
|
||||
from mcap.writer import Writer
|
||||
|
||||
args = docopt.docopt(__doc__)
|
||||
input_file = args['<csv_file>']
|
||||
output_file = args['<mcap_file>'] or input_file.replace('.csv', '.mcap')
|
||||
if input_file == output_file:
|
||||
raise ValueError('Input and output files are the same')
|
||||
|
||||
csv_file = open(input_file, 'r')
|
||||
csv_reader = csv.reader(csv_file, delimiter=',')
|
||||
header = next(csv_reader)
|
||||
|
||||
mcap_file = open(output_file, 'wb')
|
||||
writer = Writer(mcap_file)
|
||||
writer.start()
|
||||
|
||||
properties = {key: {'type': 'number'} for key in header}
|
||||
schema_id = writer.register_schema(
|
||||
name="state",
|
||||
encoding="jsonschema",
|
||||
data=json.dumps({"type": "object", "properties": properties}).encode(),
|
||||
)
|
||||
|
||||
channel_id = writer.register_channel(
|
||||
schema_id=schema_id,
|
||||
topic="state",
|
||||
message_encoding="json",
|
||||
)
|
||||
|
||||
for row in csv_reader:
|
||||
data = {key: float(value) for key, value in zip(header, row)}
|
||||
timestamp = round(float(row[0]) * 1e9)
|
||||
writer.add_message(channel_id=channel_id, log_time=timestamp, data=json.dumps(data).encode(), publish_time=timestamp,)
|
||||
|
||||
writer.finish()
|
||||
23
tools/csv_to_ulog/CMakeLists.txt
Normal file
@@ -0,0 +1,23 @@
|
||||
cmake_minimum_required(VERSION 3.15)
|
||||
project(csv_to_ulog)
|
||||
include(FetchContent)
|
||||
set(CMAKE_CXX_STANDARD 17)
|
||||
|
||||
FetchContent_Declare(
|
||||
ulog_cpp
|
||||
GIT_REPOSITORY https://github.com/PX4/ulog_cpp.git
|
||||
GIT_TAG cf24ec6
|
||||
)
|
||||
|
||||
FetchContent_Declare(
|
||||
rapidcsv
|
||||
GIT_REPOSITORY https://github.com/d99kris/rapidcsv.git
|
||||
GIT_TAG v8.82
|
||||
)
|
||||
|
||||
FetchContent_MakeAvailable(ulog_cpp)
|
||||
FetchContent_MakeAvailable(rapidcsv)
|
||||
|
||||
add_executable(csv_to_ulog csv_to_ulog.cpp)
|
||||
target_link_libraries(csv_to_ulog PUBLIC ulog_cpp::ulog_cpp)
|
||||
target_include_directories(csv_to_ulog PUBLIC ${rapidcsv_SOURCE_DIR}/src)
|
||||
20
tools/csv_to_ulog/README.md
Normal file
@@ -0,0 +1,20 @@
|
||||
# csv_to_ulog
|
||||
|
||||
Tool for converting CSV flight logs to ULog format so they can be analyzed using [FlightPlot](https://github.com/PX4/FlightPlot) software.
|
||||
|
||||
To build, go to the `<flix>/tools/csv_to_ulog` directory and run:
|
||||
|
||||
```bash
|
||||
mkdir build
|
||||
cd build
|
||||
cmake ..
|
||||
make
|
||||
```
|
||||
|
||||
Convert a CSV file to ULog:
|
||||
|
||||
```bash
|
||||
./csv_to_ulog log_file.csv
|
||||
```
|
||||
|
||||
ULog file will be saved in the same directory.
|
||||
72
tools/csv_to_ulog/csv_to_ulog.cpp
Normal file
@@ -0,0 +1,72 @@
|
||||
// Copyright (c) 2023 Oleg Kalachev <okalachev@gmail.com>
|
||||
// Repository: https://github.com/okalachev/flix
|
||||
|
||||
// Tool for conversion CSV log file to ULog format
|
||||
|
||||
#include <ulog_cpp/simple_writer.hpp>
|
||||
#include <rapidcsv.h>
|
||||
#include <vector>
|
||||
#include <string>
|
||||
#include <filesystem>
|
||||
|
||||
using std::vector;
|
||||
using std::string;
|
||||
|
||||
struct Data {
|
||||
uint64_t timestamp;
|
||||
float values[30];
|
||||
};
|
||||
|
||||
int main(int argc, char** argv)
|
||||
{
|
||||
if (argc < 2) {
|
||||
printf("Usage: %s file.csv [file.ulg]\n", argv[0]);
|
||||
return -1;
|
||||
}
|
||||
|
||||
// check input file exists
|
||||
if (!std::filesystem::exists(argv[1])) {
|
||||
printf("Input file \"%s\" does not exist\n", argv[1]);
|
||||
return -1;
|
||||
}
|
||||
|
||||
// open csv file
|
||||
rapidcsv::Document csv(argv[1]);
|
||||
auto columns = csv.GetColumnNames();
|
||||
|
||||
|
||||
// open ulog file
|
||||
string ulog_file;
|
||||
if (argc < 3) {
|
||||
ulog_file = std::filesystem::path(argv[1]).replace_extension(".ulg").string();
|
||||
} else {
|
||||
ulog_file = argv[2];
|
||||
}
|
||||
ulog_cpp::SimpleWriter writer(ulog_file.c_str(), 0);
|
||||
writer.writeInfo("sys_name", "flix");
|
||||
|
||||
vector<ulog_cpp::Field> fields;
|
||||
fields.push_back(ulog_cpp::Field("uint64_t", "timestamp"));
|
||||
columns.erase(columns.begin()); // remove timestamp column
|
||||
for (auto& column : columns) {
|
||||
// Valid field name for ULog: [a-z0-9_]+
|
||||
std::replace(column.begin(), column.end(), '.', '_'); // replace dots with underscores
|
||||
std::transform(column.begin(), column.end(), column.begin(), [](unsigned char c) { return std::tolower(c); }); // lowercase column name
|
||||
fields.push_back(ulog_cpp::Field("float", column));
|
||||
}
|
||||
|
||||
const char* msg_name = "state";
|
||||
writer.writeMessageFormat(msg_name, fields);
|
||||
writer.headerComplete();
|
||||
|
||||
const uint16_t msg_id = writer.writeAddLoggedMessage(msg_name);
|
||||
|
||||
for (size_t i = 0; i < csv.GetRowCount(); i++) {
|
||||
Data data;
|
||||
data.timestamp = csv.GetCell<float>(0, i) * 1000000.0;
|
||||
for (size_t j = 1; j <= columns.size(); j++) {
|
||||
data.values[j - 1] = csv.GetCell<float>(j, i);
|
||||
}
|
||||
writer.writeData(msg_id, data);
|
||||
}
|
||||
}
|
||||
@@ -9,8 +9,7 @@ PORT = os.environ['PORT']
|
||||
DIR = os.path.dirname(os.path.realpath(__file__))
|
||||
|
||||
dev = serial.Serial(port=PORT, baudrate=115200, timeout=0.5)
|
||||
|
||||
log = open(f'{DIR}/log/{datetime.datetime.now().isoformat()}.csv', 'wb')
|
||||
lines = []
|
||||
|
||||
print('Downloading log...')
|
||||
count = 0
|
||||
@@ -19,8 +18,14 @@ while True:
|
||||
line = dev.readline()
|
||||
if not line:
|
||||
break
|
||||
log.write(line)
|
||||
lines.append(line)
|
||||
count += 1
|
||||
print(f'\r{count} lines', end='')
|
||||
|
||||
# sort by timestamp
|
||||
header = lines.pop(0)
|
||||
lines.sort(key=lambda line: float(line.split(b',')[0]))
|
||||
|
||||
log = open(f'{DIR}/log/{datetime.datetime.now().isoformat()}.csv', 'wb')
|
||||
log.writelines([header] + lines)
|
||||
print(f'\nWritten {os.path.relpath(log.name, os.curdir)}')
|
||||
|
||||
3
tools/requirements.txt
Normal file
@@ -0,0 +1,3 @@
|
||||
docopt
|
||||
matplotlib
|
||||
mcap
|
||||