Arduino-Lite是由RoboPeak开发并使用的轻量级且高效率的AVR固件库,他基于Arduino项目的固件库改良而来,我们保留了Arduino固件库的简单易用的风格,同时有效的缩小了固件的代码尺寸,相比Arduino,Arduino-Lite拥有诸多优点。并且,他是相当易于使用的。
这里,我们很荣幸将Arduino-Lite开放下载并推荐给大家。
开放版本的Arduino-Lite可以在google code下载到: http://code.google.com/p/arduino-lite/
这是本系列文章的第一部分,后续我们将介绍Arduino-Lite的具体使用。
Arduino-Lite支持基于Atmega328/168的标准Arduino控制板外,任何第三方兼容版,任何满足下文列出的AVR芯片构成的电路均可支持。我们将在后续文章中介绍并开源RoboPeak使用的免驱动AVR/51编程器RP USB Connector,它也是采用Arduino-Lite作为固件库开发的。
同样使用C++/C编写且基于avr-gcc编译器。但与Arduino固件库相比,Arduino-Lite有如下优势。
使用Arduino-Lite的固件往往比使用Arduino固件库小了50%以上.
许多Arduino-Lite提供的与Arduino固件相同功能的函数,例如digitalWrite之Arduino-Lite版本:DIGITAL_WRITE仅使用一条AVR指令实现.
除了 Atmega8(A), Atmega168(PA), Atmega328(PA), Atmega1280 芯片外, Arduino-Lite 也支持以下芯片: Attiny2313, Attiny26, Atmega48(PA), Atmega88(PA)
对于时钟频率, Arduino-Lite 支持从1Mhz 至 20Mhz 的频率范围.
除此之外,Arduino-Lite还有如下特点:
只要系统中带有文本编辑器,即可直接用Arduino-Lite进行AVR固件开发、编译、烧录等动作。Arduino-Lite自带了avr-gcc(WINAVR)以及相关的函数库。
创建一个新的Arduino-Lite工程,最简单的办法是将模板工程文件夹解压缩并重命名为希望的工程文件。并将相关的源代码以任何目录结构放置于工程目录下,Arduino-Lite就能编译项目,无需用户修改/编写/生成Makefile.
我们认为Arduino-Lite适用于以下领域
相比Ardunio的固件库,Arduino-Lite或许不适合
Arduino项目所提供的固件库凭借其易用性被广大爱好者所接受,例如可以使用digitalWrite() digitalRead() analogWrite()函数轻松实现对AVR芯片IO口输出输入的控制。这大大降低了AVR单片机的使用门槛,使用者无需知道传统AVR编程中遇到的DDR, PIN这些寄存器,甚至像PWM输出这类需要了解定时器工作模式的功能,即使是没有任何单片机开发背景的爱好者也能迅速用Arduino固件库上手。
例如,使用传统的avr-gcc (WINAVR)进行开发时,如果实现利用PWM输出实现LED小灯的渐变点亮,开发人员需要编写至少如下代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
//Set PWM mode: fast mode for timer 0
sbi(TCCR0A, WGM01);
sbi(TCCR0A, WGM00);
//set timer 0 prescale factor to 64
sbi(TCCR0B, CS01);
sbi(TCCR0B, CS00);
sbi(DDRB, PB1);
sbi(TCCR, COM11);
unsigned
char
led_brightness = 0;
for
(led_brightness=0; led_brightness = 255; led_brightness++)
{
OCR01 = led_brightness;
}
|
上述代码要求编写者事先记住AVR相关寄存器的名称以及使用方式,这无疑加重了开发难度,即使是有一定经验的开发人员,也可能忘记某个寄存器的具体使用。另外,AVR不同型号芯片上的寄存器名称以及使用(定义)方式也是大不相同的。
而用Arduino固件库以后,上述代码则可以用如下简洁的代码所替代:
1
2
3
4
5
6
7
|
pinMode(9, OUTPUT);
unsigned
char
led_brightness = 0;
for
(led_brightness=0; led_brightness = 255; led_brightness++)
{
analogWrite(9, led_brightness);
}
|
Arduino固件库不但使得代码变得简洁易懂,同时也有使得固件代码实现了跨芯片编译。总体上说,我们认为Arduino固件库有如下优点值得学习:
基于上述优点以及Arduino硬件设计的易用性,Arduino固件也逐渐被大家所接受。Arduino-Lite同样保留了上述优点。上述Arduino代码在使用Arduino-Lite时,可以写为:
1
2
3
4
5
6
7
|
PIN_MODE(9, OUTPUT);
unsigned
char
led_brightness = 0;
for
(led_brightness=0; led_brightness = 255; led_brightness++)
{
ANALOG_WRITE(9, led_brightness);
}
|
上述代码除了大小写变化外,与Arduino固件代码一致。大部分Arduino-Lite函数均以原先Arduino对应函数的大写形式出现,方便Arduino使用者迅速迁移到Arduino-Lite。
但在提供上述优点的同时,我们发现Arduino也有不足之处
当使用digitalRead() digitalWrite() analogWrite()等函数以及依赖他们的类/函数/库时,使用者往往会发现他们的运行效果不理想。这主要与这些函数的实现有关。
因为Arduino引入了将IO引脚用数字编号的特性,在调用操作IO引脚的函数时,它们需要查表将对应的数字编号转化成对应的控制寄存器,这里以analogWrite()函数的实现为例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
|
void
analogWrite(uint8_t pin, uint8_t val)
{
pinMode(pin, OUTPUT);
if
(digitalPinToTimer(pin) == TIMER1A) {
// connect pwm to pin on timer 1, channel A
sbi(TCCR1A, COM1A1);
// set pwm duty
OCR1A = val;
}
else
if
(digitalPinToTimer(pin) == TIMER1B) {
// connect pwm to pin on timer 1, channel B
sbi(TCCR1A, COM1B1);
// set pwm duty
OCR1B = val;
}
else
if
(digitalPinToTimer(pin) == TIMER0A) {
if
(val == 0) {
digitalWrite(pin, LOW);
}
else
{
// connect pwm to pin on timer 0, channel A
sbi(TCCR0A, COM0A1);
// set pwm duty
OCR0A = val;
}
}
else
if
(digitalPinToTimer(pin) == TIMER0B) {
if
(val == 0) {
digitalWrite(pin, LOW);
}
else
{
// connect pwm to pin on timer 0, channel B
sbi(TCCR0A, COM0B1);
// set pwm duty
OCR0B = val;
}
}
else
if
(digitalPinToTimer(pin) == TIMER2A) {
// connect pwm to pin on timer 2, channel A
sbi(TCCR2A, COM2A1);
// set pwm duty
OCR2A = val;
}
else
if
(digitalPinToTimer(pin) == TIMER2B) {
// connect pwm to pin on timer 2, channel B
sbi(TCCR2A, COM2B1);
// set pwm duty
OCR2B = val;
}
else
if
(val < 128)
digitalWrite(pin, LOW);
else
digitalWrite(pin, HIGH);
}
|
代码是比较长的,相比之下,直接操作寄存器完成analogWrite同等功能,只需要如下简短的代码:
1
|
OCR01 = led_brightness;
|
如此的差距体现在执行时就是10倍以上的指令集消耗和效率的减缓。这使得Arduino固件库难以直接应用于对运行效率较高的产品和专业领域。
同样,因为Arduino固件库函数多包含大量查表代码以及需要带有引脚映射表,使得基于Arduino固件库的固件代码尺寸较大。例如之前给出的LED渐变点亮的例子,使用Arduino的固件代码在将近2kb左右。对于Atmega48/Attiny这类单片机,2kb的固件已经消耗了一半甚至所有的程序空间。
如此大的体积消耗在价格敏感产品领域是难以接受的,即使是爱好者使用,较大的固件尺寸也限制了所开发程序的功能。
目前的Arduino固件库仅支持Atmega8/Atmega168/Atmega328/Atmega1280等芯片,同时需要使用频率为8Mhz或者16Mhz的外部石英振荡器。然而,产品和业余制作中,可能会用到Atiny系列,Atmega48,Atmega88这类程序控件较少但成本更低廉的AVR型号,同时,具体应用中也会使用除了16Mhz,8Mhz之外的时钟。对于这些情况,Arduino库就无法使用了。
Arduino-Lite延续了Arduino固件库易用的特性,Arduino所使用的IO引脚编号方式仍然可以在Arduino-Lite中使用,通过,几乎所有的Arduino核心函数(如digitalRead/digitalWrite/analogWrite)都有对应的Arduino-lite函数/宏。在实现相同的易用性情况下,Arduino-lite的代码更加精简和高效。实现这一点的主要方式是采用宏展开,将Arduino固件库进行的IO数字编号->寄存器的转化任务在编译阶段完成。省去了查表操作后,固件的体积和运算速度均会显著提高。
这里我们仍旧以PWM输出为例,analogWrite(pin, value)在Arduino-lite中对应于ANALOG_WRITE(pin, value) (大部分Arduino-Lite函数均以原先Arduino对应函数的大写形式出现)。
例如给9号Arduino Pin口 (PB1)输出PWM信号,可以写为
1
|
ANALOG_WRITE(9, pwm_value);
|
这虽然形式上和用法上和Arduino固件库一致,但是Arduino-Lite会将ANALOG_WRITE进行宏展开:
1
2
3
4
5
6
|
#define ANALOG_WRITE( pin, val ) \
do
{ \
PWM_ENABLE(pin); \
PWM_SET(pin,val); \
} \
while
(0)
|
而PWM_ENABLE和PWM_SET宏最终会展开为:
1
2
|
sbi(TCCR1A, COM1A1);
OCR1A = pwm_value;
|
相比上面给出的长长地Arduino实现,Arduino-Lite仅产生2条AVR指令即可完成完全相同的任务。
并且,Arduino-Lite也提供了更加精简的接口,如PWM_SET(pin, value)完成PWM输出的控制,使用这类接口时,Arduino-Lite仅产生一条AVR指令。
这里我们用产生固件的尺寸做个直观对比
实现LED渐亮效果(PWM)
- Arduino库产生的最终代码: 2048 byte
- Arduino-Lite产生的最终代码: 100byte
另外,在Arduino固件库中,ADC引脚(PC)均只能供analogRead使用,无法用做传统的IO口。Arduino-Lite中,我们扩充了Arduino的引脚编号,将这些ADC引脚也纳入了普通IO控制范围,共开发者调用DIGITAL_READ/WRITE操作。如下是AtemgaX8(m48,m88,m168,m328)的Arduino-Lite引脚定义:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
|
// ATMEL ATMEGA8 & 168
//
// +-\/-+
// PC6 1| |28 PC5 (AI 5/*D19)
// (D 0) PD0 2| |27 PC4 (AI 4/*D18)
// (D 1) PD1 3| |26 PC3 (AI 3/*D17)
// (D 2) PD2 4| |25 PC2 (AI 2/*D16)
// PWM+ (D 3) PD3 5| |24 PC1 (AI 1/*D15)
// (D 4) PD4 6| |23 PC0 (AI 0/*D14)
// VCC 7| |22 GND
// GND 8| |21 AREF
// *(D20) PB6 9| |20 AVCC
// *(D21) PB7 10| |19 PB5 (D 13)
// PWM+ (D 5) PD5 11| |18 PB4 (D 12)
// PWM+ (D 6) PD6 12| |17 PB3 (D 11) PWM
// (D 7) PD7 13| |16 PB2 (D 10) PWM
// (D 8 )PB0 14| |15 PB1 (D 9) PWM
// +----+
|
在下篇文章中,我们将详细介绍Arduino-Lite的安装、使用过程,以及如果用它来给AVR/Arduino设备编程。以及相关的函数介绍