51单片机(STC89C52)在Ubuntu下的开发
简介
都是8051衍生的8位单片机, STC单片机有89/90/10/11/12/15这几个大系列, 每个系列的特点如下
- 89系列是传统的8051单片机, 烧录方法有区别, 但是功能上可以和AT89系列兼容, 属于12T单片机
- 90系列是89系列的改进型, 12T单片机
- 10和11系列是1T单片机, 有PWM, 4态IO接口, EEPROM等功能, 但都没有ADC
- 12系列是增强型功能的1T单片机, 型号后缀
AD
的带ADC, 后缀S2
的除了ADC还带双串口 - 15系列是最新的产品, 内部集成了高精度R/C时钟, 不需要外部晶振
STC89C52参数
STC89C52和STC89C51的区别仅在于Flash大小, C51为4K而C52为8K. 都有多种封装, 常见的是体积较大的DIP40宽体双列
- Flash: 8K bytes
- RAM: 512 bytes
- 内置4KB EEPROM
- 32-bit I/O
- 看门狗定时器, MAX810复位电路
- 3个16-bit定时器
- 4个外部中断, 一个7向量4级中断结构和全双工串行口
- 最高运作频率35MHz, 6T/12T可选
- VCC是5V, 不要高于这个电压.
如果需要ADC, 可以选 STC12C5A60S2 系列, 1T型的指令时钟速度比普通STC89系列(12T)快, 做定时时要注意.
硬件准备
- 使用常见的C51最小开发板
- 一个USB2TTL的转接卡
- 用于查看输出的LED+1K限流电阻
在Ubuntu20.04下的开发
软件部分
- 编译工具 SDCC http://sdcc.sourceforge.net/
- 烧录工具 stcgal https://github.com/grigorig/stcgal
- (可选)封装库 HML_FwLib_STC89 https://github.com/MCU-ZHISHAN-IoT/HML_FwLib_STC89
安装sdcc
Ubuntu20.04自带的sdcc版本为3.8.0, 所以打算自行编译安装. 先阅读sdcc的安装说明, 在 http://sdcc.sourceforge.net/doc/sdccman.pdf
- 前往 https://sourceforge.net/projects/sdcc/files/sdcc/4.1.0/ 下载最新的4.1.0源码.
- 解压
tar -xvjf sdcc-src-4.1.0.tar.bz2
- 配置
./configure
, 过程中提示gputils需要安装, 通过apt install gputils
安装, 再次./configure
未再报错, 但是提示这个gputils比较旧, 有些设备不支持, 先忽略 - 编译
make
, 过程中提示缺texinfo, 通过apt install texinfo
安装, 然后再make. 需要说一下的是编译时间很长, 如果仅仅需要编译C51, 可以在配置时禁掉其他一些不需要的芯片. - 安装
sudo make install
检查和测试安装好的sdcc
- 执行
sdcc --version
看看输出是否正确, 执行sdcc --print-search-dirs
查看库文件的位置. 默认的8051的库文件位置为/usr/local/share/sdcc/include/mcs51/
- 写一段代码test001.c,
char test;
void main(void) {
test = 0;
}
用sdcc编译, 如果生成结果文件正常没有报错, 就说明安装没问题
# 这一步会产生test001.asm等文件
sdcc -c test001.c
# 这一步会产生test001.ihx等文件
sdcc test001.c
安装stcgal
根据Github上项目首页的说明, 直接用pip3 install stcgal
安装, 然后执行stcgal -h
查看输出
编写代码并用sdcc编译
编写一个test002.c, 通过控制占空比让P0.7上的LED产生呼吸灯的效果.
- 8052.h是sdcc自带的通用的C52头文件, 不是STC89Cxx的定制头文件
- 需要STC的头文件, 对应STC89C52, 可以用stc89.h; 对应STC12C5A60S2, 可以用stc12.h
#include <8052.h>
#define Led10 P0_7
typedef unsigned int u16;
int atime = 64;
void delay(u16 pms) {
u16 x, y;
for (x=pms; x>0; x--) {
for (y=11; y>0; y--);
}
}
void ledfade(u16 i) {
Led10 = 0;
delay(i);
Led10 = 1;
delay(atime-i);
}
int main(void) {
u16 a, b;
while(1) {
for (a=0; a<atime; a++) {
for (b=0; b < (atime - a)/4; b++) {
ledfade(a);
}
}
for (a=atime; a>0; a--) {
for (b=0; b < (atime - a)/4; b++) {
ledfade(a);
}
}
}
}
编译有两种方式
- 加上
-m<port>
参数, 对应89C52/12C5A60S2, 用-mmcs51
, 如果是Makefile, 加到CCFLAG里面 - 不加参数, 在头文件中指定路径, 例如
#include <mcs51/8052.h>
这是使用参数的方式
sdcc -mmcs51 test002.c
烧录
烧录使用stcgal, 这个页面下有详细的使用方法
对于STC89C52使用-P stc89
, 对于STC12C5A60S2系列, 使用-P stc12
stcgal -P stc89 test002.ihx
Waiting for MCU, please cycle power: done # 这里要关闭再打开电源
Target model:
Name: STC89C52RC/LE52R
Magic: F002
Code flash: 8.0 KB
EEPROM flash: 6.0 KB
Target frequency: 11.010 MHz
Target BSL version: 3.2C
Target options:
cpu_6t_enabled=False
bsl_pindetect_enabled=False
eeprom_erase_enabled=False
clock_gain=high
ale_enabled=True
xram_enabled=True
watchdog_por_enabled=False
Loading flash: 243 bytes (Intel HEX)
Switching to 19200 baud: checking setting testing done
Erasing 2 blocks: done
Writing flash: 640 Bytes [00:00, 1844.10 Bytes/s]
Setting options: done
Disconnected!
烧录完就会自动运行. 默认的波特率为19200, 写入较慢, 对于STC89C52RC可以直接使用115200的波特率, 用-b
参数指定
stcgal -P stc89 -b 115200 test002.ihx
如果同时有多个com口, 用-p
指定端口
stcgal -P stc89 -b 115200 dist/89/89.hex -p /dev/ttyUSB1
使用-D
参数能显示出串口的交互信息
stcgal -D -P stc89 -b 115200 test002.ihx
Waiting for MCU, please cycle power: <- Packet data: 46 B9 68 00 1C 00 0A 76 0A 72 0A 76 0A 72 0A 76 0A 72 0A 76 0A 72 32 43 FD F0 02 82 5A 16
done
Target model:
Name: STC89C52RC/LE52R
Magic: F002
Code flash: 8.0 KB
EEPROM flash: 6.0 KB
Target frequency: 11.010 MHz
Target BSL version: 3.2C
Target options:
cpu_6t_enabled=False
bsl_pindetect_enabled=False
eeprom_erase_enabled=False
clock_gain=high
ale_enabled=True
xram_enabled=True
watchdog_por_enabled=False
Loading flash: 337 bytes (Intel HEX)
Switching to 115200 baud: checking -> Packet data: 46 B9 6A 00 0C 8F FF FD 00 06 A0 81 28 16
<- Packet data: 46 B9 68 00 0C 8F FF FD 00 06 A0 81 26 16
setting -> Packet data: 46 B9 6A 00 0B 8E FF FD 00 06 A0 A5 16
<- Packet data: 46 B9 68 00 0B 8E FF FD 00 06 A0 A3 16
testing -> Packet data: 46 B9 6A 00 0C 80 00 00 36 01 F0 02 1F 16
<- Packet data: 46 B9 68 00 06 80 EE 16
-> Packet data: 46 B9 6A 00 0C 80 00 00 36 01 F0 02 1F 16
<- Packet data: 46 B9 68 00 06 80 EE 16
-> Packet data: 46 B9 6A 00 0C 80 00 00 36 01 F0 02 1F 16
<- Packet data: 46 B9 68 00 06 80 EE 16
-> Packet data: 46 B9 6A 00 0C 80 00 00 36 01 F0 02 1F 16
<- Packet data: 46 B9 68 00 06 80 EE 16
done
Erasing 2 blocks: -> Packet data: 46 B9 6A 00 0D 84 02 33 33 33 33 33 33 2F 16
<- Packet data: 46 B9 68 00 06 80 EE 16
done
-> Packet data: 46 B9 6A 00 8C 00 00 00 00 00 00 80 02 00 06 02 00 AC 75 81 09 12 01 4D E5 82 60 03 02 00 03 79 00 E9 44 00 60 1B 7A 00 90 01 51 78 01 75 A0 00 E4 93 F2 A3 08 B8 00 02 05 A0 D9 F4 DA F2 75 A0 FF E4 78 FF F6 D8 FD 78 00 E8 44 00 60 0A 79 01 75 A0 00 E4 F3 09 D8 FC 78 00 E8 44 00 60 0C 79 00 90 00 01 E4 F0 A3 D8 FC D9 FA 75 08 40 75 09 00 02 00 03 AE 82 AF 83 EE 4F 60 14 7C 0B 7D 00 1C BC FF 01 1D EC 4D 70 F7 1E BE FF FB 16
<- Packet data: 46 B9 68 00 07 80 85 74 16
Writing flash: 0%| | 0/512 [00:00<?, ? Bytes/s]-> Packet data: 46 B9 6A 00 8C 00 00 00 00 80 00 80 01 1F 80 E8 22 AE 82 AF 83 C2 87 8E 82 8F 83 C0 07 C0 06 12 00 68 D0 06 D0 07 D2 87 AC 08 AD 09 EC C3 9E F5 82 ED 9F F5 83 02 00 68 7E 00 7F 00 AC 08 AD 09 C3 EE 9C EF 9D 50 45 7C 00 7D 00 AA 08 AB 09 EA C3 9E FA EB 9F C3 13 CA 13 CA C3 13 CA 13 CA FB C3 EC 9A ED 9B 50 1E 8E 82 8F 83 C0 07 C0 06 C0 05 C0 04 12 00 85 D0 04 D0 05 D0 06 D0 07 0C BC 00 C9 0D 80 C6 0E BE 00 B3 0F 80 B0 14 16
<- Packet data: 46 B9 68 00 07 80 1E 0D 16
-> Packet data: 46 B9 6A 00 8C 00 00 00 01 00 00 80 AE 08 AF 09 EE 4F 60 A4 7C 00 7D 00 AA 08 AB 09 EA C3 9E FA EB 9F C3 13 CA 13 CA C3 13 CA 13 CA FB C3 EC 9A ED 9B 50 1E 8E 82 8F 83 C0 07 C0 06 C0 05 C0 04 12 00 85 D0 04 D0 05 D0 06 D0 07 0C BC 00 C9 0D 80 C6 1E BE FF 01 1F 80 B7 75 82 00 22 FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF AD 16
<- Packet data: 46 B9 68 00 07 80 36 25 16
Writing flash: 75%|█████████████████████████▌ | 384/512 [00:00<00:00, 3814.67 Bytes/s]-> Packet data: 46 B9 6A 00 8C 00 00 00 01 80 00 80 FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF 77 16
<- Packet data: 46 B9 68 00 07 80 80 6F 16
Writing flash: 640 Bytes [00:00, 4207.06 Bytes/s]
Setting options: -> Packet data: 46 B9 6A 00 0A 8D FD FF FF FF FB 16
<- Packet data: 46 B9 68 00 0A 8D FD FF FF FF F9 16
done
-> Packet data: 46 B9 6A 00 06 82 F2 16
Disconnected!
SDCC 下的串口
开启串口, 并将putchar()重定向到串口
- 对于使用11.0592晶振的STC89C52
- 使用T1,不加倍时最高波特率只能达到28800, 加倍后是57600
- 使用T2, T2作为波特率发生器时, 递增频率为晶振频率的2分频, 可以达到115200
- 对于使用11.0592晶振的STC12C5A60S2, 1T模式波特率达到115200没问题, 如果开了12T模式, 和89C52是一样的
头文件serial.h
#ifndef __SERIAL_H__
#define __SERIAL_H__
#include <mcs51/8051.h>
void serial_init(void);
#endif // __SERIAL_H__
C文件serial.c
#include "serial.h"
// configure serial for 9600 baud, 8 data bits, 1 stop bit.
void serial_init(void) {
TMOD = 0x21;
SCON = 0x40;
TH1 = 0xFD; // TH1 = 256 - 11.0592 * 1000 * 1000 / 12 / 32 / 9600;
TCON |= 0x40; // start the timer1
SCON |= 0x02; // tell putchar() the serial is ready to send
}
如果使用T2, 波特率公式如下:
波特率 = 11059200 / { 32×[65536-(RCAP2H,RCAP2L)] }
其中的RCAP2H,RCAP2L为自动重装值,由上式得
RCAP2H, RCAP2L = 65536 - 11059200 / (32×波特率)
这样得波特率为115200时,RCAP2H, RCAP2L = 0xff, 0xfd
void init_com( void ) {
SCON=0x50; //串口工作方式1,8位UART,波特率可变
TH2=0xFF;
TL2=0xFD; //波特率:115200 晶振=11.0592MHz
RCAP2H=0xFF;
RCAP2L=0xFD; //16位自动再装入值
TCLK=1; //波特率发生器工作方式
RCLK=1;
C_T2=0;
EXEN2=0;
TR2=1 ; //定时器2开始
}
SDCC下的时钟中断
用11.0592MHz晶振的C52产生较精确的1秒定时中断, 定时器初始值由如下计算得到
- 由晶振11.0592 MHz, 得到定时器时钟为 11.0592 / 12 = 0.9216 MHz,
- 因此1ms对应 921.6 个时钟周期,
- 因此50ms对应 46080 个时钟周期,
- 将其设为一次中断后, 20次中断就对应1s
代码
#include <8052.h>
const unsigned char th = (65536 - 46080) / 256;
const unsigned char tl = (65536 - 46080) % 256;
volatile unsigned char i = 0;
void main() {
TMOD= 0x01; //工作方式为16位定时器
TH0 = th; //计数寄存器高8位
TL0 = tl; //计数寄存器低8位
EA = 1; //允许中断
ET0 = 0x01; //允许T0中断
TR0 = 1; //启动T0
while(1);
}
void Timer0IRQ(void) __interrupt (1) // 中断处理函数 T0 -> 中断1
{
i++;
if(i > 20) {
P0_7 = (P0_7 == 1)? 0 : 1; //触发P0.7 LED闪烁
i = 1; // 注意这边不能初始化为0, 否则每次会多跑一个中断
}
TH0 = th; //计数寄存器高8 位重新载入
TL0 = tl; //计数寄存器低8 位重新载入
}
参考
- Very helpful SDCC C51 code examples https://github.com/hungtcs-lab/8051-examples
51单片机(STC89C52)在Ubuntu下的开发的更多相关文章
- ubuntu下Nodic开发环境搭建
ubuntu下Nodic开发环境搭建 1.编译环境 ubuntu可直接装gcc编译环境 sudo apt install gcc-arm-none-eabi 也可以下载可执行文件download 2. ...
- 在Ubuntu下爽快开发Android必要的5款装备
每一个程序员都有一颗极客的心,一些小装备肯定就比不可少啦.我刚刚从windows中转到Ubuntu,除了要适应ubuntu外,也想将windows中用惯了的小软件一起搬过去.在这里简单地罗列一下自己在 ...
- Ubuntu下Java开发环境搭建(eclipse)
最近把工作环境转移到了Ubuntu Kylin下,发现在这下面Java环境还是很方便的.然而也经历了一些摸索的过程,故作文以记之. 一/开发前准备 安装系统/配置软件源,这部分内容没什么需要注意的.O ...
- Ubuntu 下python开发环境的搭建
一.安装python3 ubuntu自身是安装python2的,例如在ubuntu 16.04中安装的就是python2.7.但我想在python3的环境下进行开发所以就要安装python3.但由于u ...
- 【Linux/Ubuntu学习5】Ubuntu 下android 开发,eclipse不能识别手机
ubuntu下eclipse不能识别手机解决方法: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 1) 在终端运行 lsusb 会发现结果 ...
- 安装Ubuntu下的开发工具
刚安装好的Ubuntu,还缺乏很多开发工具.这些工具都可以通过网络进行安装. 1. 更新软件源$ sudo apt-get update 2.安装.配置.启动ftp服务.执行以下命令安装,安装后即会自 ...
- [转]Ubuntu下ROS开发环境搭建(QT+ros_qtc_plugin)
ROS与C++入门教程-搭建开发环境(QT+ros_qtc_plugin) PS : 在“安装ros_qtc_plugin插件”这一步中,原文提到“ Ubuntu 14.04使用apt-get方式安装 ...
- 重要:Linux下IDE--KDevelop (用来跟踪调试C++) Ubuntu下QT4开发环境的搭建及初体验
Linux下安装Qt4有两大问题,一是环境变量,二是IDE(集成开发环境).安装Qt4也有两种方法,一种是apt-get,一种是下载源码包,而后 一种方法已经人证实是最有可能不好使的方法.所以我最终采 ...
- 打造Ubuntu下Java开发环境
一.了解JDK 不同的java软件和类库对jdk有不同要求,在了解如何安装Java之前,让我们快速地了解JRE.OpenJDK和Oracle JDK之间的不同之处. JRE(Java Runtime ...
- ubuntu下android开发环境安装
一 安装jdk 网址:http://www.oracle.com/technetwork/java/javase/downloads/jdk7-downloads-1880260.html 新建一个文 ...
随机推荐
- 01-UPF介绍
Mainstream Low Power techniques clock gating - 低功耗的设计风格,4bit以上可以通过综合工具增加ICG APR的时候可以更换器件VT Advanced ...
- 非标准库--conio.h库
1.getch函数 主要内容 int getch(void): 所在头文件:conio.h 函数用途:从控制台读取一个字符,但不显示在屏幕上,即一个不需要通过ENTER确定的getchar. 函数原型 ...
- window-子系统-ubuntu
window-子系统-ubuntu 1. 背景 提供类Linux开发环境(命令行.文件系统.进程管理.网路) 2. 安装 A. wsl 安装 下载链接: https://wslstorestorage ...
- Linux-用户组-groupad-groupdel-usermod
- [转帖]TiFlash 源码阅读(一) TiFlash 存储层概览
https://cloud.tencent.com/developer/article/1988629 背景 本系列会聚焦在 TiFlash 自身,读者需要有一些对 TiDB 基本的知识.可以通过这三 ...
- [转帖]oswbb工具分析主机性能
https://www.cnblogs.com/lkj371/p/15154268.html 在进行数据库故障分析和数据库例行扩容评估时,需要对数据库主机的CPU.内存.磁盘.网络进行负荷分析,常规处 ...
- [转帖]“高密度核心”的角逐 —— AMD Bergamo SoC & Zen 4c 前瞻
https://zhuanlan.zhihu.com/p/585469720 最近这段时间一直在关注Bergamo和Zen 4c,但遗憾的是Genoa的发布会并没有提到太多的Bergamo细节.不过这 ...
- [转帖]Nginx-https证书认证详解
https://developer.aliyun.com/article/885650?spm=a2c6h.24874632.expert-profile.306.7c46cfe9h5DxWK 简介: ...
- 一种轻量分表方案-MyBatis拦截器分表实践
背景 部门内有一些亿级别核心业务表增速非常快,增量日均100W,但线上业务只依赖近一周的数据.随着数据量的迅速增长,慢SQL频发,数据库性能下降,系统稳定性受到严重影响.本篇文章,将分享如何使用MyB ...
- It is currently in use by another Gradle instance
FAILURE: Build failed with an exception. * What went wrong: Could not create service of type TaskHis ...