找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1625|回复: 0
打印 上一主题 下一主题
收起左侧

如何根据《Keyer自动键控制单片机程序》反推出电路原理图 求帮助

[复制链接]
跳转到指定楼层
楼主
ID:573727 发表于 2019-6-29 13:50 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
  1. /*******************************************************************************
  2. 程序名称:Keyer 自动键控制程序
  3. *******************************************************************************/
  4. #include <STC89C5xRC.H>
  5. #include <intrins.h>
  6. #include <stdlib.h>
  7. #include <config.h>
  8. /* 定义IO端口宏 */
  9. #define GPIO_DIG P0 //定义数码管IO
  10. #define GPIO_KEY P1 //定义键盘IO
  11. /* 定义数码管 */
  12. sbit LSA = P2^2;
  13. sbit LSB = P2^3;
  14. sbit LSC = P2^4;
  15. /* 定义IO端口 */
  16. sbit BEEP  = P1^5; //定义蜂鸣器
  17. sbit CWOUT = P2^0; //定义电键信号发送端口与LED
  18. sbit K1 = P3^0;  //定义K1键为功能键
  19. sbit K2 = P3^1;  //定义K2键为播放键
  20. sbit K3 = P3^2;  //定义Dot键使用外部中断INT0
  21. sbit K4 = P3^3;  //定义Dash键使用外部中断INT1
  22. /* 声明功能函数 */
  23. void IntConfiguration(void);  //中断配置
  24. void Delay(uint t);     //延时(t=毫秒数)
  25. void TimeCount(void);    //计算Dot/Dash时长
  26. void Paddle(void);     //自控电码发生
  27. void Straight(void);    //手控电码发生
  28. void Space(uchar n);    //间隔发生器(n=间隔数目)
  29. void Function(void);    //功能键控制
  30. void PlayCall(void);    //发送本台呼号
  31. void Display(void);     //数码管显示
  32. void RecordInfor(void);    //录入本台呼号
  33. void WriteSystem(void);    //写入系统参数
  34. void Player(uchar m, uchar x[]); //发送内置信息(m=发送字符数)
  35. /* 声明ISP操作函数 */
  36. void EEPROM_Read(uint EE_address, uchar *DataAddress, uchar n);  //字节读
  37. void EEPROM_Write(uint EE_address, uchar *DataAddress, uchar n); //字节写
  38. void EEPROM_SectorErase(uint EE_address);       //扇区擦除
  39. /* 数码管显示码表,0 1 2 3 4 5 6 7 8 9 */
  40. const uchar code DIG_CODE[10] = {0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07, 0x7F, 0x6F};
  41. /* 侧音频率表,400,500,600,700,800,900,1000Hz */
  42. const uchar code Tone_H[7] = {0xFB, 0xFC, 0xFC, 0xFD, 0xFD, 0xFD, 0xFE};   
  43. const uchar code Tone_L[7] = {0x1E, 0x18, 0xBF, 0x36, 0x8F, 0xD4, 0x0C};
  44. /* 呼叫数组 */
  45. const uchar code C1[10] = {'1','0','1','0','.','1','1','0','1',' '};
  46. const uchar code C2[6]  = {'1','0','0','.','0',' '};
  47. const uchar code C3[14] = {'0','1','1','0','.','0','0','0','.','0',' ','1','0','1'};
  48. /* 定义变量 */
  49. uint DotTime, DashTime;  //Dot时长,Dash时长
  50. uint TCount, Tmax;   //时长计数,时长阈值
  51. uchar KeyValue;    //功能键值
  52. uchar Cmax = 54, CW_x[56]; //数组上标,录入缓存
  53. uchar WPM = 16;    //发报速度4-60wpm (默认16wpm)
  54. uchar ToneFreq = 3;   //侧音频率序号0-6 (默认3/700Hz)
  55. uchar Number;    //播发字符数
  56. bit KeyFlg  = 1;   //点划交替标志(1=无点划交替)
  57. bit SendFlg = 0;   //发送标志(1=禁止发射)
  58. static uint T2Count = 0; //定时器T2计数变量
  59. void main(void)
  60. {
  61. /* 设定初始值 */
  62. IntConfiguration(); //配置中断
  63. if (K1 == 0)  //按K1键开机进行系统重置
  64. {
  65.   WPM = 16;
  66.   ToneFreq = 3;
  67.   Number = 0;
  68.   WriteSystem();
  69.   EEPROM_SectorErase(SectorAddr_2);
  70. }
  71. else
  72. {
  73.   EEPROM_Read(SectorAddr_1, CW_x, 3); //读系统设定参数
  74.   if (CW_x[0] != 0xFF)
  75.   {
  76.    WPM = CW_x[0];
  77.    ToneFreq = CW_x[1];
  78.    Number = CW_x[2];
  79.   }
  80. }
  81. TimeCount();   //计算Dot/Dash时长
  82. TH1 = Tone_H[ToneFreq]; //设定定时器T1(侧音频率)初始值
  83.     TL1 = Tone_L[ToneFreq];
  84. TR2 = 1;    //开定时器T2
  85. /* 运行按键监测程序 */
  86. while (1)
  87. {
  88.   SendFlg = 0; //允许发射
  89.   Display();
  90.   Delay(15);
  91.   if (K1 == 0) //功能键按下
  92.   {
  93.    Delay(1000);
  94.    if (K1 == 0)
  95.     Function();
  96.   }
  97.   if (K2 == 0 && Number > 0)
  98.    PlayCall();
  99. }
  100. }
  101. /*******************************************************************************
  102. 函 数 名:IntConfiguration
  103. 函数功能:设置中断参数
  104. *******************************************************************************/
  105. void IntConfiguration(void)
  106. {
  107. IT0 = 0;  //INT0为低电平触发方式
  108. IT1 = 0;  //INT1为低电平触发方式
  109. EX0 = 1;  //INT0中断允许
  110. EX1 = 1;   //INT1中断允许
  111. TMOD = 0x12; //定时器T0工作方式为2,定时器T1工作方式为1
  112. PT0 = 1;  //定时器T0(时长)高优先级
  113. PT1 = 1;  //定时器T1(侧音)高优先级
  114. ET0 = 1;  //定时器T0中断允许
  115. ET1 = 1;  //定时器T1中断允许
  116. T2CON = 0x00; //定时器T2工作方式为16位自动重载定时器
  117. ET2 = 1;  //定时器T2中断允许
  118. TH0 = 0x9C;  //设定定时器T0为100us
  119.     TL0 = 0x9C;
  120. RCAP2H = 0x0B; //设定定时器T2为每秒中断16次
  121.     RCAP2L = 0xDC;
  122. EA = 1;   //开总中断
  123. }
  124. /*******************************************************************************
  125. 函 数 名:Int0
  126. 函数功能:外部中断0触发时发送Dot
  127. *******************************************************************************/
  128. void Int0(void) interrupt 0
  129. {
  130. Delay(5); //消抖
  131. if (K3 == 0)
  132. {
  133.   if (KeyFlg == 1) //不是点划交替发送Dot
  134.    {SEND_DOT();}
  135.   if (K4 == 0)  //点划交替发送Dash
  136.   {
  137.    SEND_DASH();
  138.    KeyFlg = 1;  //清除点划交替标志
  139.   }
  140. }
  141. }
  142. /*******************************************************************************
  143. 函 数 名:Int1
  144. 函数功能:外部中断1触发时发送Dash
  145. *******************************************************************************/
  146. void Int1(void) interrupt 2
  147. {
  148. Delay(5); //消抖
  149. if (K4 == 0)
  150. {
  151.   SEND_DASH();   
  152.   if (K3 == 0) //点划交替发送Dot
  153.   {
  154.    SEND_DOT();   
  155.    KeyFlg = 0; //设置点划交替标志
  156.   }
  157. }
  158. }
  159. /*******************************************************************************
  160. 函 数 名:Timer0
  161. 函数功能:定时器T0的中断函数,产生100us/次的延时,控制发码时长。
  162. *******************************************************************************/     
  163. void Timer0(void) interrupt 1
  164. {
  165. TCount++;
  166. }
  167. /*******************************************************************************
  168. 函 数 名:Timer1(void)
  169. 函数功能:定时器T1的中断函数,控制侧音频率,ToneFre为侧音频率的序列号。
  170. *******************************************************************************/     
  171. void Timer1(void) interrupt 3
  172. {
  173. BEEP = ~BEEP;
  174. TH1 = Tone_H[ToneFreq];
  175.     TL1 = Tone_L[ToneFreq];
  176. }
  177. /*******************************************************************************
  178. 函 数 名:Timer2
  179. 函数功能:定时器T2的中断函数。
  180. *******************************************************************************/
  181. void Timer2(void) interrupt 5
  182. {
  183. TF2 = 0;
  184. T2Count++;
  185. }
  186. /*******************************************************************************
  187. 函 数 名:Delay
  188. 函数功能:1ms延时函数
  189. 输    入:t 延时时长 t x 1ms
  190. *******************************************************************************/
  191. void Delay(uint t)
  192. {
  193. uint max = 10 * t;  //计算时长阈值
  194. TCount = 0;
  195. TR0 = 1;
  196. while (TCount <= max); //产生延时
  197. TR0 = 0;
  198. }
  199. /*******************************************************************************
  200. 函 数 名:TimeCount
  201. 函数功能:计算Dot/Dash时长
  202. *******************************************************************************/
  203. void TimeCount(void)
  204. {
  205. DotTime = 12000 / WPM;
  206. DashTime = 3 * DotTime;
  207. }
  208. /*******************************************************************************
  209. 函 数 名:Paddle
  210. 函数功能:产生自动键电码
  211. *******************************************************************************/
  212. void Paddle(void)
  213. {
  214. uint max = Tmax; //设定时长阈值
  215. TR1 = 1;   //开音频振荡器
  216. TCount = 0;   //时长计数器清零
  217. CWOUT = SendFlg; //设定发射状态
  218. TR0 = 1;   //开时长计数器
  219. while (TCount <= max); //发送  
  220. TR0 = 0; //关时长计数器
  221. CWOUT = 1; //禁止发射
  222. TR1 = 0; //关音频振荡器
  223. Space(1); //点划间隔
  224. T2Count = 0;
  225. }
  226. /*******************************************************************************
  227. 函 数 名:Space
  228. 函数功能:产生间隔
  229. 输    入:n 间隔数量
  230. *******************************************************************************/
  231. void Space(uchar n)
  232. {
  233. uint max = n * DotTime; //计算时长阈值
  234. TCount = 0;    //时长计数器清零
  235. TR0 = 1;    //开时长计数器
  236. while (TCount <= max); //产生间隔
  237. TR0 = 0;    //关时长计数器
  238. }
  239. /*******************************************************************************
  240. 函 数 名:Function
  241. 函数功能:按照功能键执行相关设置
  242. *******************************************************************************/
  243. void Function(void)
  244. {
  245. bit p = 0;
  246. INT_OFF();  //关INT
  247. SendFlg = 1; //禁止发射
  248. KeyValue = 0;
  249. while (K1 == 0) //长按键,每1秒变换一个功能
  250. {
  251.   Delay(100);    //每100ms计数一次
  252.   if (++KeyValue > 40) //保持循环按键
  253.    KeyValue = 0;
  254.   if (KeyValue % 10 == 0) //到达变换时间
  255.   {
  256.    switch (KeyValue)
  257.    {
  258.     case 10: SEND_DASH(); break; //设定发报速度提示
  259.     case 20: SEND_DASH(); SEND_DASH(); break; //设定侧音频率提示
  260.     case 30: SEND_DASH(); SEND_DASH(); SEND_DASH(); break; //录入呼号提示
  261.    }
  262.   }
  263. }
  264. KeyValue /= 10;
  265. T2Count = 0;
  266. switch (KeyValue)
  267. {        
  268.   case 1: //调整发报速度
  269.    while (K1 == 1 && T2Count <= 96) //K1按下或6秒内无键按下则退出
  270.    {
  271.     Display();
  272.     if (K4 == 0 && WPM < 60) //WPM+
  273.     {
  274.      while (T2Count < 16 && K4 == 0); //松手延迟
  275.      WPM++;
  276.      TimeCount();
  277.      SEND_DASH();
  278.      p = 1;
  279.     }
  280.     if (K3 == 0 && WPM > 4) //WPM-
  281.     {
  282.      while (T2Count < 16 && K3 == 0 ); //松手延迟
  283.      WPM--;
  284.      TimeCount();
  285.      SEND_DASH();
  286.      p = 1;
  287.     }
  288.    }
  289.    if (p == 1)
  290.     WriteSystem(); //记忆参数
  291.    SEND_DASH();
  292.    break;
  293.   case 2: //设定侧音频率
  294.    while (K1 == 1 && T2Count <= 96) //K1按下或6秒内无键按下则退出
  295.    {
  296.     Display();
  297.     if (K3 == 0 && ToneFreq > 0) //频率减按下
  298.     {
  299.      while (T2Count < 16 && K3 == 0); //松手延迟
  300.      ToneFreq--;
  301.      TH1 = Tone_H[ToneFreq];
  302.      TL1 = Tone_L[ToneFreq];
  303.      SEND_DASH();
  304.      p = 1;
  305.     }
  306.     if (K4 == 0 && ToneFreq < 6) //频率加按下
  307.     {
  308.      while (T2Count < 16 && K4 == 0); //松手延迟
  309.      ToneFreq++;
  310.      TH1 = Tone_H[ToneFreq];
  311.      TL1 = Tone_L[ToneFreq];
  312.      SEND_DASH();
  313.      p = 1;
  314.     }
  315.    }
  316.    if (p == 1)
  317.     WriteSystem(); //记忆参数
  318.    SEND_DASH();
  319.    break;
  320.   case 3: //录入呼号
  321.    RecordInfor();
  322. }
  323. INT_ON(); //开INT
  324. }
  325. /*******************************************************************************
  326. 函 数 名:RecordInfor
  327. 函数功能:录入内置信息,最多10个字符
  328. *******************************************************************************/
  329. void RecordInfor(void)
  330. {
  331. bit k = 0;  //间隔标志清零
  332. Number = 0;  //字符数清零     
  333. T2Count = 0; //退出计时清零
  334. while (K1 && Number < Cmax && T2Count <= 320) //按K1键/点划数为最大值/者20秒无按键则退出
  335. {
  336.   if (!K3) //Dot键按下
  337.   {     
  338.    Delay(5); //消抖
  339.    if (!K3)
  340.    {
  341.     CW_x[Number++] = '0';
  342.     SEND_DOT();
  343.     TCount = 0; //间隔计时清零
  344.     TR0 = 1; //开定时器T0用于添加字符间隔
  345.     k = 1;  //设置间隔标志
  346.    }
  347.   }
  348.   if (!K4) //Dash键按下
  349.   {
  350.    Delay(5);
  351.    if (!K4)
  352.    {
  353.     CW_x[Number++] = '1';
  354.     SEND_DASH();
  355.     TCount = 0;
  356.     TR0 = 1;
  357.     k = 1;
  358.    }
  359.   }
  360.   if (k && TCount >= DashTime) //添加字符间隔
  361.   {
  362.    CW_x[Number++] = '.';
  363.    TR0 = 0; //关定时器T0
  364.    k = 0;  //清除间隔标志
  365.   }
  366. }
  367. if (Number > 0) //有信息录入则补字符间隔、记忆字符数
  368. {
  369.   if (CW_x[Number - 1] != '.')
  370.    CW_x[Number] = '.' ;
  371.   WriteSystem();
  372. }
  373. }
  374. /*******************************************************************************
  375. 函 数 名:WriteSystem
  376. 函数功能:写入系统参数
  377. *******************************************************************************/
  378. void WriteSystem(void)
  379. {
  380. EEPROM_SectorErase(SectorAddr_1);
  381. CW_x[0] = WPM;
  382. CW_x[1] = ToneFreq;
  383. CW_x[2] = Number;
  384. EEPROM_Write(SectorAddr_1, CW_x, 3);
  385. }
  386. /*******************************************************************************
  387. 函 数 名:Player
  388. 函数功能:发送内置信息
  389. 输    入:m 发送字符数,x[]信息
  390. *******************************************************************************/
  391. void Player(uchar m, uchar x[])
  392. {
  393. uchar j = 0;
  394. for (; j < m; j++)
  395.   switch (x[j])
  396.   {
  397.    case '0': SEND_DOT(); break; //发送Dot
  398.    case '1': SEND_DASH(); break; //发送Dash
  399.    case '.': Space(2); break;  //字符间隔
  400.    case ' ': Space(6);    //单词间隔
  401.   }
  402. }
  403. /*******************************************************************************
  404. 函 数 名:PlayCall
  405. 函数功能:发送呼叫
  406. *******************************************************************************/
  407. void PlayCall(void)
  408. {
  409. uchar i;
  410. EEPROM_Read(SectorAddr_2, CW_x, Number);
  411. for (i = 1; i < 4; i++)
  412.   Player(10, C1);
  413. Player(6, C2);
  414. for (i = 1; i < 4; i++)
  415. {
  416.   Player(Number, CW_x);
  417.   Space(4);
  418. }
  419. Player(14, C3);
  420. }
  421. /*******************************************************************************
  422. 函 数 名:Display
  423. 函数功能:使用数码管显示
  424. *******************************************************************************/
  425. void Display(void)
  426. {
  427. uchar i;
  428. uint f = 400 + ToneFreq * 100;
  429. for (i = 1; i < 9; i++)
  430. {
  431.   switch (i)
  432.   {
  433.    case 1: if (WPM/10 != 0) {LSC = 1; LSB = 1; LSA = 1; GPIO_DIG = DIG_CODE[WPM / 10];} break; //显示WPM十位
  434.    case 2: LSC = 1; LSB = 1; LSA = 0; GPIO_DIG = DIG_CODE[WPM % 10]; break;     //显示WPM个位
  435.    case 5: if (f == 1000) {LSC = 0; LSB = 1; LSA = 1; GPIO_DIG = 0x06;} break;  //显示侧音频率千位     
  436.    case 6: LSC = 0; LSB = 1; LSA = 0; GPIO_DIG = DIG_CODE[(f / 100) % 10]; break; //显示侧音频率百位
  437.    case 7: LSC = 0; LSB = 0; LSA = 1;GPIO_DIG = 0x3F; break; //显示侧音频率十位
  438.    case 8: LSC = 0; LSB = 0; LSA = 0;GPIO_DIG = 0x3F;   //显示侧音频率个位
  439.   }
  440.   Delay(1);
  441.   GPIO_DIG = 0x00; //消隐
  442. }
  443. }
  444. /*******************************************************************************
  445. 函 数 名:EEPROM_SectorErase
  446. 函数功能:擦除一个扇区(512字节/扇区)
  447. 输    入:扇区地址 EE_address
  448. *******************************************************************************/
  449. void EEPROM_SectorErase(uint EE_address)
  450. {
  451. EA = 0;       //关总中断
  452. ISP_ADDRH = EE_address / 256; //扇区地址高字节
  453. ISP_ADDRL = EE_address % 256; //扇区地址低字节
  454. ISP_ENABLE();     //允许ISP操作
  455. ISP_ERASE();     //扇区擦除命令
  456. ISP_TRIG();      //触发ISP操作
  457. ISP_DISABLE();     //禁止ISP操作
  458. EA = 1;       //开总中断
  459. }
  460. /*******************************************************************************
  461. 函 数 名:EEPROM_Read
  462. 函数功能:读n个字节函数(最多255字节/次)
  463. 输    入:扇区地址 EE_address, 读出数据 *DataAddress, 读出字节数 n
  464. *******************************************************************************/
  465. void EEPROM_Read(uint EE_address, uchar *DataAddress, uchar n)
  466. {
  467. EA = 0;        //关总中断
  468. ISP_ENABLE();      //允许ISP操作
  469. ISP_READ();       //字节读命令
  470. do
  471. {
  472.   ISP_ADDRH = EE_address / 256; //地址高字节
  473.   ISP_ADDRL = EE_address % 256; //地址低字节
  474.   ISP_TRIG();      //触发ISP操作
  475.   _nop_();
  476.   *DataAddress = ISP_DATA;  //读数据到指定变量
  477.   EE_address++;     //下一个地址
  478.   DataAddress++;     //下一个数据
  479. }
  480. while (--n);      //直到结束
  481. ISP_DISABLE();      //禁止ISP操作
  482. EA = 1;        //开总中断
  483. }
  484. /*******************************************************************************
  485. 函 数 名:EEPROM_Write
  486. 函数功能:写n个字节数据(最多255字节/次)
  487. 输    入:扇区地址 EE_address,写入数据 *DataAddress,写入字节数 n
  488. *******************************************************************************/
  489. void EEPROM_Write(uint EE_address, uchar *DataAddress, uchar n)
  490. {
  491. EA = 0;        //关总中断
  492. ISP_ENABLE();      //允许ISP操作
  493. ISP_WRITE();      //字节写命令
  494. do
  495. {
  496.   ISP_ADDRH = EE_address / 256; //送地址高字节
  497.   ISP_ADDRL = EE_address % 256; //送地址低字节
  498.   ISP_DATA  = *DataAddress;  //送数据到ISP_DATA
  499.   ISP_TRIG();      //触发ISP操作
  500.   _nop_();
  501.   EE_address++;     //下一个地址
  502.   DataAddress++;     //下一个数据
  503. }
  504. while (--n);      //直到结束
  505. ISP_DISABLE();      //禁止ISP操作
  506. EA = 1;        //开总中断
  507. }
复制代码


分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏1 分享淘帖 顶 踩
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表