版权声明:本文为博主[原创]文章,未经博主允许可以转载,注明博客出处:[http://blog.csdn.net/FreeApe]

目录(?)[+]

  1. 相关知识了解

    1. 1 Nuttx系统
    2. 2 NSH
  2. 编译固件和刷固件
  3. 调试方式
    1. 1 测试小功能程序
    2. 2 固件测试
  4. 调试实例演示

1 相关知识了解

1.1 Nuttx系统

  嵌入式实时操作系统(RTOS)。强调标准兼容和小型封装,具有从8位到32位微控制器环境的高度可扩展性。NuttX 主要遵循 Posix 和 ANSI 标准,对于在这些标准下不支持的功能,或者不适用于深度嵌入环境的功能(如 fork()),采用来自 Unix 和常见 RTOS (如 VxWorks)的额外的标准 API。
  支持文件系统、设备驱动、网络、USB支持、Flash支持、图形支持等。

1.2 NSH

  NuttShell和Unix终端命令类似。NSH通过串口或者USB转串口来与PX4FMU交互,因此可以使用类似超级终端的串口软件来与FMU交互,在Pixhawk开发中,建立好了开发环境后(安装了工具链等),PX4 Toolchain 已经附带了一个串口工具:TeraTerm。如果版本低了,可以单独去下载这个软件并安装。
  在PX4FMU中,NSH默认是通过USB转串口和串口5(波特率为57600)来交互。可以更改默认设置。
  将Pixhawk通过USB转串口连接上电脑上后,再打开TeraTerm软件。通过输入『?』或『help』指令可以查看当前NSH支持的指令已经编译好的应用。其他的一些指令跟Unix终端中用法差不多,不过一些指令的参数都没有了。

  通过串口5的交互能打印PX4FMU启动时的一些信息。这个可以用来查看系统启动方面的信息。当然启动完后,还可以输入一些指令来与PX4FMU来交互。


2 编译固件和刷固件

  刷固件的方式很多,可以通过地面站软件QGC或者MP都可以刷固件,可以刷稳定版的固件或者自己编译出来的自定义固件。这里通过PX4 Console来编译固件和刷固件。打开控制台,进入到固件目录:

cd /d/px4/firmware

  删除所有编译的文件,包括编译的操作系统,即删除Archives文件夹和Build文件夹里面的内容一般不使用:

make distclean

  编译操作系统:

make archives

  一般只有当Nuttx配置改变了或者submodule(在GIT中链接外部库,比如MAVLINK或者Nuttx OS)发生了改变才会去编译操作系统,平时编译一次就好了,而且这个编译一次是需要很长的时间的,一个小时左右吧。尽管并行编译会加快速度,但是配置不好,并行编译很容易在系统运行的过程中出问题。如果你使用苹果系统或者Linux系统,这个时间也会大大降低。
  删除编译的固件相关文件,即删除Build文件夹里面的内容:

make clean

  编译固件,以px4fmu-v2_default版本为例:

make px4fmu-v2_defaul

  编译并刷固件,编译完后紧接着刷固件:

make upload px4fmu-v2_default

  在控制台上用到上面的指令就差不多了。

3 调试方式

3.1 测试小功能程序

  因为在windows系统上编译一次固件需要3分钟左右吧,对于一般的功能测试程序可以通过在本机安装上VC6.0或VS或者MinGW(gcc,g++,推荐),编写测试程序编译调试。

3.2 固件测试

  通过printf打印相关信息,断点标识等。这个需要编译固件,并刷固件。Pixhawk通过USB转串口线连接电脑,用TeraTerm输入指令来调试。

4 调试实例演示

  前面已经说了通过串口5(波特率57600)可以查看系统启动打印的信息。通过USB转串口在NSH终端(TeraTerm)输入指令控制Nuttx中的应用或者一些命令。
  可以先阅读sdlog2日志记录自动清理功能分析与实现,了解sdlog2应用是怎么运行的。sdlog2应用随着系统启动时默认启动了的,当我们修改了固件中sdlog2应用的代码时,比如简单的,我们知道sdlog2_main这个函数是在后台运行的,它是检测sdlog2应用的参数输入,比如stop,start,on,off。我们也可以自己再加一条参数检测到这个函数中:

<code class="language-c++ hljs scss has-numbering">    <span class="hljs-comment">/*添加测试代码start...*/</span><span class="hljs-function">if(!<span class="hljs-function">strcmp(argv[<span class="hljs-number">1</span>],<span class="hljs-string">"hello"</span>)</span>)</span>{<span class="hljs-function">printf(<span class="hljs-string">"[YCM]hello world!\n"</span>)</span>;}<span class="hljs-comment">/*添加测试代码end...*/</span></code><ul class="pre-numbering" style="display: block;"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li></ul>
<code class="language-c++ hljs cpp has-numbering"><span class="hljs-keyword">int</span> sdlog2_main(<span class="hljs-keyword">int</span> argc, <span class="hljs-keyword">char</span> *argv[])
{<span class="hljs-keyword">if</span> (argc < <span class="hljs-number">2</span>) {sdlog2_usage(<span class="hljs-string">"missing command"</span>);<span class="hljs-keyword">return</span> <span class="hljs-number">1</span>;}<span class="hljs-keyword">if</span> (!<span class="hljs-built_in">strcmp</span>(argv[<span class="hljs-number">1</span>], <span class="hljs-string">"start"</span>)) {<span class="hljs-keyword">if</span> (thread_running) {warnx(<span class="hljs-string">"already running"</span>);<span class="hljs-comment">/* this is not an error */</span><span class="hljs-keyword">return</span> <span class="hljs-number">0</span>;}main_thread_should_exit = <span class="hljs-keyword">false</span>;deamon_task = px4_task_spawn_cmd(<span class="hljs-string">"sdlog2"</span>,SCHED_DEFAULT,SCHED_PRIORITY_DEFAULT - <span class="hljs-number">30</span>,<span class="hljs-number">3000</span>,sdlog2_thread_main,(<span class="hljs-keyword">char</span> * <span class="hljs-keyword">const</span> *)argv);<span class="hljs-keyword">return</span> <span class="hljs-number">0</span>;}<span class="hljs-keyword">if</span> (!<span class="hljs-built_in">strcmp</span>(argv[<span class="hljs-number">1</span>], <span class="hljs-string">"stop"</span>)) {<span class="hljs-keyword">if</span> (!thread_running) {warnx(<span class="hljs-string">"not started"</span>);}main_thread_should_exit = <span class="hljs-keyword">true</span>;<span class="hljs-keyword">return</span> <span class="hljs-number">0</span>;}<span class="hljs-comment">/*添加测试代码start...*/</span><span class="hljs-keyword">if</span>(!<span class="hljs-built_in">strcmp</span>(argv[<span class="hljs-number">1</span>],<span class="hljs-string">"hello"</span>)){<span class="hljs-built_in">printf</span>(<span class="hljs-string">"[YCM]hello world!\n"</span>);}<span class="hljs-comment">/*添加测试代码end...*/</span><span class="hljs-keyword">if</span> (!thread_running) {warnx(<span class="hljs-string">"not started\n"</span>);<span class="hljs-keyword">return</span> <span class="hljs-number">1</span>;}<span class="hljs-keyword">if</span> (!<span class="hljs-built_in">strcmp</span>(argv[<span class="hljs-number">1</span>], <span class="hljs-string">"status"</span>)) {sdlog2_status();<span class="hljs-keyword">return</span> <span class="hljs-number">0</span>;}<span class="hljs-keyword">if</span> (!<span class="hljs-built_in">strcmp</span>(argv[<span class="hljs-number">1</span>], <span class="hljs-string">"on"</span>)) {<span class="hljs-keyword">struct</span> vehicle_command_s cmd;cmd.command = VEHICLE_CMD_PREFLIGHT_STORAGE;cmd.param1 = -<span class="hljs-number">1</span>;cmd.param2 = -<span class="hljs-number">1</span>;cmd.param3 = <span class="hljs-number">1</span>;orb_advertise(ORB_ID(vehicle_command), &cmd);<span class="hljs-keyword">return</span> <span class="hljs-number">0</span>;}<span class="hljs-keyword">if</span> (!<span class="hljs-built_in">strcmp</span>(argv[<span class="hljs-number">1</span>], <span class="hljs-string">"off"</span>)) {<span class="hljs-keyword">struct</span> vehicle_command_s cmd;cmd.command = VEHICLE_CMD_PREFLIGHT_STORAGE;cmd.param1 = -<span class="hljs-number">1</span>;cmd.param2 = -<span class="hljs-number">1</span>;cmd.param3 = <span class="hljs-number">2</span>;orb_advertise(ORB_ID(vehicle_command), &cmd);<span class="hljs-keyword">return</span> <span class="hljs-number">0</span>;}sdlog2_usage(<span class="hljs-string">"unrecognized command"</span>);<span class="hljs-keyword">return</span> <span class="hljs-number">1</span>;
}
</code><ul class="pre-numbering" style="display: block;"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li><li>31</li><li>32</li><li>33</li><li>34</li><li>35</li><li>36</li><li>37</li><li>38</li><li>39</li><li>40</li><li>41</li><li>42</li><li>43</li><li>44</li><li>45</li><li>46</li><li>47</li><li>48</li><li>49</li><li>50</li><li>51</li><li>52</li><li>53</li><li>54</li><li>55</li><li>56</li><li>57</li><li>58</li><li>59</li><li>60</li><li>61</li><li>62</li><li>63</li><li>64</li><li>65</li><li>66</li><li>67</li><li>68</li><li>69</li><li>70</li><li>71</li><li>72</li><li>73</li><li>74</li></ul>

  编译固件并刷固件,如果你的代码有问题,那么会报错,终止编译,会有错误提示:

make upload px4fmu-v2_default

  从编译过程中打印信息,我们也可以知道这个submodule中什么改变了,就需要重新编译操作系统了。编译完,刷好了固件:


  再次连接TeraTerm,输入指令:

sdlog2 hello

  输出hello world!了。

Pixhawk---基于NSH的Firmware开发与调试相关推荐

  1. PixHawk飞控和Mission Planner地面站安装调试

    PixHawk飞控和Mission Planner地面站安装调试 PixHawk是著名飞控厂商3DR推出的新一代独立.开源.高效的飞行控制器,前身为APM飞控,不仅提供了丰富的外设模块和可靠的飞行体验 ...

  2. 基于FPGA的以太网开发

      基于FPGA的以太网开发,在调试过的FPGA玩家开来,其实算不上很难的技术!但是如果只是菜鸟级别的选手,没有调试过的话,就有些头疼了!早在自己在实习的时候,就接触到XAUI(万兆以太网口)接口,但 ...

  3. 基于IAR上搭建开发MM32的环境

    简 介: 初步搭建了MindMotion基于IAR的软件开发环境.测试了基于DAPLink对于SeekFree开发板的开发功能.但是对于测试实验板,DAPLink无法寻找到目标MCU. 利用MM32- ...

  4. java osgi web开发_基于 OSGi 和 Spring 开发 Web 应用

    开发一个简单的OSGi Web应用实例 一个简单的Web应用 我们写一个简单的 Web 应用 compute.html :计算两个数字的和或乘积.如下图所示: 图1.一个简单例子 一个简单例子.bmp ...

  5. 使用VS Code 从零开始开发并调试.NET Core 应用程序

    使用VS Code 从零开始开发并调试.NET Core 应用程序,C#调试.上一篇 使用VS Code开发 调试.NET Core 应用程序 得到了大家的支持. 现在为大家带来从零开始教程,让你更好 ...

  6. 基于ROS的移动机器人开发:视觉、语音、导航

    经过十余年的快速发展,ROS已成为当下主流的移动机器人操作系统,被各大机器人科技企业支持与使用.一些重大的自动驾驶或者机器人开源框架都是基于ROS系统进行开发的,例如:百度的Apllo和日本的Auto ...

  7. 使用Android Studio 进行NDK开发和调试

    2019独角兽企业重金招聘Python工程师标准>>> 尽管Android Studio已经越来越流行了,但很多人还是习惯于Eclipse或源码环境下开发JNI应用.个人认为使用An ...

  8. 基于IDEA的JavaWeb开发环境搭建

    基于IDEA的JavaWeb开发环境搭建 基于IDEA的JavaWeb开发环境搭建 jdk下载安装配置环境变量 下载 安装 配置环境变量 下载安装激活使用IntelliJ IDEA 下载 安装 激活 ...

  9. AngularJS学习笔记之二:开发、调试和测试工具

    2019独角兽企业重金招聘Python工程师标准>>> 一.搭建自动化的前端开发.调试和测试环境 我们先来看一个完整的项目实例,这是AngularJS官方为我们提供的Phonecat ...

最新文章

  1. c++ 函数指针_开发经验分享(5) 修改Makefile实现C/C++混合编程
  2. 食出100分:‘粥’的做法4---鱼片瘦肉粥
  3. request 和response
  4. samba升级_潮闻快食 | adidas Originals经典鞋款Samba进化升级,C.E x Nike联名系列全释出!...
  5. 初学者如何学习Vim
  6. Vue实现仿音乐播放器7-实现音乐榜单效果
  7. python3 -c 和 python3 -m
  8. linux gnome 桌面,GNOME Linux桌面
  9. 判断C语言变量名是否合法
  10. (一)Multisim安装与入门
  11. y的花式写法_26字母的花式写法可复制 也叫圆体字写法如下一基本规则
  12. fpga加载程序慢_FPGA JTAG接口下载速度很慢
  13. 推流式搅拌器选型功率计算方法_不同池形中推流搅拌器功率消耗的数值模拟
  14. 推荐系统相关顶会整理
  15. Android之讯飞语音-文字转语音(不用另外安装语音合成包apk)遇到的问题
  16. MyEclipse全局搜索
  17. 【操作系统】第五章——虚拟内存技术
  18. 天工开物 #5 我的 Linux 开发机
  19. pca图解读_利用R绘制PCA分析图(2)
  20. 电路的耦合方式 直接耦合、阻容耦合、变压器耦合 光电耦合。

热门文章

  1. spark.yarn.archive spark.yarn.jars
  2. vue 关于图片和文字的绝对定位 js 动态设置定位
  3. 智慧执行 大数据的重塑之功
  4. 搭建mysql 主从复制The slave I/O thread stops because master and slave have equal MySQL server UUIDs
  5. react大数据量渲染_React大量数据渲染的绝佳解决方案——React虚拟化组件
  6. 全面概述将人工智能市场纳入其广泛的数据库
  7. 没有比这个更详细的Elasticsearch教程
  8. 怎么把pdf电子书转换成txt格式
  9. Jupyter Notebook基础(6)Jupyter Notebook命令行命令帮助
  10. jQuery淡入浅出