Arduino平台基于DbC的软件调试
基于LED和串口通信的DBC调试工具:HAssert --- Hyper LED/Serial Assert 。
本文基于DbC思想 ,在Arduino平台上实现了两种断言显示方式---LED显示和串口输出显示。
LED显示方式
适用于没有串口(这种情况很少),或者串口已经被占用的情况,只用一个LED的显示次数来调试程序;
串口输出显示方式
可以把断言内容传送到PC机,在PC机看到出错的位置(文件名和行号)。
以下的HAssert.h和HAssert.cpp实现了DbC调试的断言。
一.HAssert.h头文件
#ifndef H_ASSERT_H
#define H_ASSERT_H
#include <stdint.h> // C99-standard exact-width integers
#include <avr/pgmspace.h>
//#define ROM_BYTE(rom_var_) pgm_read_byte_near(&(rom_var_))
//#define ROM PROGMEM
/// 1.有/无断言选择开关(配置1)
// (1)调试时, H_ASSERT = 1,断言有效
// (2)发布软件时,H_ASSERT = 0,自动去掉代码中插入的所有断言
#define H_ASSERT_EN 1
///2.LED/串口显示断言选择开关(配置2)
//(1)H_LED_ASSERT_EN = 1时,选择LED显示断言
//(2)H_LED_ASSERT_EN = 0时,选择串口显示断言
#define H_LED_ASSERT_EN 0
//=================================================
/// 状态LED引脚 (配置3)
#define H_ASSERT_LED_PIN 13 //Arduino板用13; MSR和SWT板用RxErr状态LED,7.
/// 状态LED延时时间(配置4)
// (1)仿真用了10000ms, 实际上可以修改成1秒的时间
#define H_ASSERT_LED_DELAY_TIME 1000 // ms
//================================================
///3.H_ASSERT_EN == 0,去掉断言宏
#if H_ASSERT_EN == 0
#define H_ASSERT_ID(id, test)
#define H_REQUIRE_ID(id, test)
#define H_ERROR_ID(id)
#define H_BREAKPOINT_ID(id)
#define H_DEFINE_THIS_FILE
#define H_DEFINE_THIS_MODULE(name_)
#define H_ASSERT(test)
#define H_REQUIRE(test)
#define H_ERROR()
#define H_BREAK_POINT()
///4.H_ASSERT_EN == 1,加入断言宏
#else
///4.0.声明断言函数
#ifdef __cplusplus
extern "C" {
#endif
void H_onAssert_id(uint8_t id); //LED显示断言回调函数声明
void H_onAssert(char const PROGMEM * const file, int line); //串口显示断言回调函数声明
#ifdef __cplusplus
}
#endif
///4.1.LED断言宏
#if H_LED_ASSERT_EN == 1
///LED断言---带id参数
//(1).id参数可以指示错误分类等,如用LED闪亮的次数分类
//(2). 建议id = 2~255, id=0和1都算做1次,以下类同
///断言宏
#define H_ASSERT_ID(id, test) if (test) { \
} \
else (H_onAssert_id((id)))
///必要条件
#define H_REQUIRE_ID(id, test) H_ASSERT_ID(id, test)
///错误
#define H_ERROR_ID(id) (H_onAssert_id((id)))
///断点
#define H_BREAKPOINT_ID(id) (H_onAssert_id((id)))
///4.2.串口断言宏
#else
///加入到每个文件的#include "HSAssert.h"之后,文件的前边。
//绝对路径文件,太长!
#define H_DEFINE_THIS_FILE \
static char const PROGMEM l_file[] = __FILE__;
///自己写文件名,或模块名
#define H_DEFINE_THIS_MODULE(module_name_) \
static char const PROGMEM l_file[] = #module_name_;
///断言宏
#define H_ASSERT(test) if (test) {\
} \
else (H_onAssert(l_file, __LINE__))
///必要条件
#define H_REQUIRE(test) H_ASSERT(test)
///错误
#define H_ERROR() (H_onAssert(l_file, __LINE__))
///断点 停止
#define H_BREAKPOINT() { Serial.print("--- Breakpoint --- ");\
(H_onAssert(l_file, __LINE__)); \
cli();\
while(1);\
} while(0)
#endif
#endif
#endif
二.HAssert.cpp文件
#include <Arduino.h>
#include "HAssert.h"
///1.LED断言函数实现---显示闪亮次数
//(1).用一个LED亮表示有错,闪亮次数表示错误类型
//(2).有代码中不同的位置放id不同的断言宏,用闪亮的次数表示错误类型
//(3).建议id = 2~255,不建议用0和1.
//(4).有错误就会停止
void H_onAssert_id(uint8_t id) {
pinMode(H_ASSERT_LED_PIN, OUTPUT);
sei();
delay(H_ASSERT_LED_DELAY_TIME);
for (uint8_t i = 0; i < id; i++) {
digitalWrite(H_ASSERT_LED_PIN, LOW);
delay(H_ASSERT_LED_DELAY_TIME);
digitalWrite(H_ASSERT_LED_PIN, HIGH);
delay(H_ASSERT_LED_DELAY_TIME);
}
cli();
while(1) { ; } //停止,断言不会返回,错误就得处理掉!
//asm volatile ("jmp 0x0000"); // 复位
}
///2.串行断言函数实现---显示文件和出错行
//只有H_BREAKPOINT()会停止
void H_onAssert(char const PROGMEM * const file, int line) {
int i =0 ;
char ch, buffer[60];
do {
ch = (char)pgm_read_word(file + i);
buffer[i++] = ch;
} while((ch != 0)&&(i < 60) );
Serial.print(line - 3,DEC); Serial.print(" Line Error in ");
Serial.print(buffer); Serial.println(" file.");
}
三.DbC测试程序
1.利用LED的DbC测试
//////////////////////////////////////////////////////////////////////////////
// 名称:利用LED的DbC测试
//////////////////////////////////////////////////////////////////////////////
///LED断言要求:HAssert.h中
//配置1:H_ASSERT_EN = 1
//配置2:H_LED_ASSERT_EN = 1
#include "HAssert.h"
void setup() {
}
void loop() {
uint8_t x = 5;
uint8_t y = 6;
H_REQUIRE_ID(2,x==y); //判断逻辑错误时,闪亮2次,最后保持亮
H_REQUIRE_ID(3,x != y); //判断逻辑正确时,不亮
H_ERROR_ID(10); //闪亮10次,最后保持亮
H_BREAKPOINT_ID(10);//与 H_ERROR_ID(10)一样
delay(100);
while(1);
}
2.利用串口通信的DbC测试
//////////////////////////////////////////////////////////////////////////////
// 名称:利用串口通信的DbC测试
//////////////////////////////////////////////////////////////////////////////
///串口断言要求:HAssert.h中
//配置1:H_ASSERT_EN = 1
//配置2:H_LED_ASSERT_EN = 0
#include "HAssert.h"
#include <Arduino.h>
H_DEFINE_THIS_MODULE("Test_HAssert.ino") //自己写文件名
//HS_DEFINE_THIS_FILE //绝对路径文件,太长!不用
void setup() {
Serial.begin(9600);
}
void loop() {
uint8_t x = 5;
uint8_t y = 6;
H_ASSERT(false);
H_REQUIRE(x == y);
H_REQUIRE(x != y); //满足条件,不输出
H_ERROR();
H_BREAKPOINT();
delay(100);
//while(1);
}
四.串口通信DbC测试效果
本文中只给出串口DbC测试的运行效果,LED的DbC测试是在Arduino板上用LED闪亮次数来表示出错的位置。
Arduino平台基于DbC的软件调试的更多相关文章
- 《软件调试的艺术》学习笔记——GDB使用技巧摘要
<软件调试的艺术>学习笔记——GDB使用技巧摘要 <软件调试的艺术>,因为名是The Art of Debugging with GDB, DDD, and Eclipse. ...
- 洪强宁:宜信PaaS平台基于Calico的容器网络实践
洪强宁:宜信PaaS平台基于Calico的容器网络实践 本文内容来自由七牛云主办的ECUG Con,独家授权InfoQ整理完成 容器云面临的网络挑战 在传统的IDC的架构里面网络是很重要的事情,在 ...
- <读书笔记>软件调试之道 :从大局看调试-零容忍策略
声明:本文档的内容主要来源于书籍<软件调试修炼之道>作者Paul Butcher,属于读书笔记.欢迎转载! ---------------------------------------- ...
- 10月12号 晚八点 Speed-BI 云平台-基于Excel数据源的管理驾驶舱构建全过程,腾讯课堂开课啦
认真地做了一大摞一大摞的报表,老板没时间看?努力把能反馈的内容都融汇进图表里,老板嫌复杂?做了几个简单的报表,老板一眼就觉得信息不全面?每个报表都用了各种各样的图表,老板却毫无兴趣?明明很努力了,为什 ...
- PfSense基于BSD的软件防火墙的安装、配置与应用
PfSense基于BSD的软件防火墙的安装.配置与应用 PfSense是一个FreeBSD下的免费开源的防火墙和路由器软件,他为了在X86平台上面建立一个高集成性的防火墙项目,下面就为大家展示如何配置 ...
- 【小梅哥FPGA进阶教程】第九章 基于串口猎人软件的串口示波器
九.基于串口猎人软件的串口示波器 1.实验介绍 本实验,为芯航线开发板的综合实验,该实验利用芯航线开发板上的ADC.独立按键.UART等外设,搭建了一个具备丰富功能的数据采集卡,芯航线开发板负责进行数 ...
- centos平台基于snort、barnyard2以及base的IDS(入侵检测系统)的搭建与测试及所遇问题汇总
centos平台基于snort.barnyard2以及base的IDS(入侵检测系统)的搭建与测试及所遇问题汇总 原创 2016年12月19日 01:20:03 标签: centos / snort ...
- 基于Azure的软件部署和开发系列沙龙
活动简介: Azure是一种灵活和支持互操作的平台,它可以被用来创建云中运行的应用或者通过基于云的特性来加强现有应用.它开放式的架构给开发者提供了Web应用.互联设备的应用.个人电脑.服务器.或者提供 ...
- .net平台 基于 XMPP协议的即时消息服务端简单实现
.net平台 基于 XMPP协议的即时消息服务端简单实现 昨天抽空学习了一下XMPP,在网上找了好久,中文的资料太少了所以做这个简单的例子,今天才完成.公司也正在准备开发基于XMPP协议的即时通讯工具 ...
随机推荐
- win7下tomcat5.5无法通过ip和127.0.0.1访问的解决方法
解决办法:找到tomcat5.5目录下的conf\server.xml文件,原文如下: <Connector port="8080" maxHttpHeaderSize=&q ...
- 常见WEB开发安全漏洞 原因分析及解决
目 录 1 会话标识未更新 3 1.1 原因 3 1.2 解决 3 2 SQL注入 3 2.1 原因 3 2.2 解决 5 3 XSS跨站脚本编制 5 3.1 原因 5 3.2 解决 5 4 XSRF ...
- Oracle彻底杀掉进程
kill session 是DBA经常碰到的事情之一.如果kill 掉了不该kill 的session,则具有破坏性,因此尽可能的避免这样的错误发生.同时也应当注意,如果kill 的session属于 ...
- March 7 2017 Week 10 Tuesday
Age is a very high price to pay for maturity. 年纪是成熟的代价. A high price, indeed a high price. It is bes ...
- ZT C++ 重载、覆盖和隐藏的区别
重载.覆盖和隐藏的区别 分类: C++ 学习笔记 学习心得与方法 2013-09-26 11:21 50人阅读 评论(0) 收藏 举报 概念区分 “overload”翻译过来就是:超载,过载,重载,超 ...
- 【转】应用程序的入口是ActivityThread
ActivityThread运行框架 在分析中,我们可以看到真正对应应用进程的不是Application而是ActivityThread.我们从实际的应用堆栈可以看到: NavitiveStart.m ...
- SSH Secure Shell Client连接Linux断开
修改/etc/ssh/sshd_config文件,将 ClientAliveInterval 0和ClientAliveCountMax 3的注释符号去掉,将ClientAliveInterval对应 ...
- 【luogu P4711 「化学」相对分子质量】 题解
题目链接:https://www.luogu.org/problemnew/show/P4711 要细心模拟 #include <cstdio> #include <algorith ...
- Ueditor上传图片到本地改造到上传图片到七牛云存储
作为新手说多了都是泪啊!我特别想记录一下作为菜鸟时的坑.看看以后是否会看着笑出来. 为什么要改到云存储上就不说了.好处多多. 视频教程上使用的又拍云同时也提到了七牛云.下来我自己也查了下.又拍云是试用 ...
- U盘空间释放
U盘做成启动盘后,怎么释放空间,恢复到原来的容量.这里有个帖子,我觉得很方便,不用安装什么东西,简单直接. http://blog.sina.com.cn/s/blog_68f6e8a901014cv ...