久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

帖子
查看: 4827|回復: 5
打印 上一主題 下一主題
收起左側

用stc單片機制作PC流光溢彩 源程序

  [復制鏈接]
跳轉到指定樓層
樓主
ID:687694 發表于 2020-10-19 20:13 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
近日在B站看到許多PC流光溢彩效果,如:https://www.bilibili.com/video/BV1fJ411B7bp?from=search&seid=16736407731500993634還有https://www.bilibili.com/video/BV1MC4y1h7iB?from=search&seid=16736407731500993634
效果是真不錯,還有許多制作的視頻帖子,但是。。。。。。都是基于arduino制作的,我手頭上只有一塊自制的arduino uno,個頭有點大,洞洞板上還沒有usb轉串口,有點麻煩。正好上次制作rgb環形時鐘的硬件符合條件,板上60顆燈珠,stc主控,帶usb轉串口,那么我們就用這個stc來做流光溢彩吧。
查了一下網上也沒有stc的教程,就自己分析了一下運行過程。
首先由PC機運行AmbiBox軟件抓取屏幕點,取得rgb顏色數據。
然后由串口發送至單片機解析。
解析后的數據刷新至rgbled顯示。
過程貌似比較簡單,但我們得分析其中的協議。
網查資料得到 Ada模式協議如下:
前三個字節分別是Ada,然后3個字節的前2個是led個數,最后一個是校驗碼,接著就是對應個數的rgb數據,每個led3字節,分別對應r,g,b。
有了這個就簡單了。構建程序流程如下
單片機主頻33.1776(為了配合串口波特率)
啟動程序->初始化串口(定時器2,波特率115200,)->初始化led數組 ->檢測顯示(紅,綠,藍各顯示一遍)->開啟串口接收中斷->主循環判斷如果不允許中斷則刷新rgbled,刷新完成后開啟中斷,如果允許中斷則等待不允許中斷。
中斷函數中流程->依次判斷接收字符是否對應,如果前3個字節對應則 接收電燈數據,并將數據保存至數組中,接收一組數據完畢,失能中斷,這樣不會因中斷接收數據影響刷新。
圖我也懶得拍照了,程序上傳上來,大家想做的可以自己畫個板子,燒上代碼就行了,至于AmbiBox自行某度即可。

單片機源程序如下:
  1. #include "stc15.h"
  2. #include "ws2812.h"
  3. uchar xdata rgb[60][3];//led數組
  4. uchar sendrgb=0;//預發送rgb變量
  5. uchar h=0,l=0,o=0;
  6. uchar in=0;
  7. void delayms(int x)
  8. {
  9.         int i,j;
  10.         for(i=0;i<x;i++)
  11.         for(j=0;j<1000;j++);
  12. }

  13. void UartInit(void)                //115200bps@33.1776MHz
  14. {
  15.         SCON = 0x50;                //8位數據,可變波特率
  16.         AUXR |= 0x01;                //串口1選擇定時器2為波特率發生器
  17.         AUXR |= 0x04;                //定時器2時鐘為Fosc,即1T
  18.         T2L = 0xB8;                //設定定時初值
  19.         T2H = 0xFF;                //設定定時初值
  20.         AUXR |= 0x10;                //啟動定時器2
  21.         ES = 1;              //使能串口1中斷
  22. }
  23. void initrgb(uchar st,uchar sp,uchar rg0,uchar rg1,uchar rg2)//起始位,結束位,rgb1,2,3(初始化數組)
  24. {
  25. uchar i,j;
  26.         for(i=st;i<sp;i++)
  27.         {               
  28.                 for(j=0;j<3;j++)
  29.                 {
  30.                         switch(j)
  31.                         {
  32.                                 case 0:
  33.                         rgb[i][j]=rg0;
  34.                                 break;
  35.                                 case 1:
  36.                         rgb[i][j]=rg1;
  37.                                 break;
  38.                                 case 2:
  39.                         rgb[i][j]=rg2;
  40.                                 break;
  41.                         }
  42.                 }
  43.         }       
  44. }

  45. void display()     //刷新rebled輸出函數
  46. {
  47. uchar i,j,k;
  48.         for(i=0;i<60;i++)
  49.         {
  50.                 for(j=0;j<3;j++)
  51.                 {
  52.                         sendrgb=rgb[i][j];
  53.                         for(k=0;k<8;k++)
  54.                         {
  55.                                 if(sendrgb&0x80)
  56.                                         st1();
  57.                                 else
  58.                                         st0();
  59.                                 sendrgb<<=1;
  60.                         }
  61.                 }
  62.         }
  63.         stop();
  64. }

  65. void main()
  66. {
  67. UartInit();//初始化串口
  68. initrgb(0,60,50,0,0);        //測試綠
  69. display();
  70. delayms(1000);
  71. initrgb(0,60,0,50,0);        //測試紅
  72. display();
  73. delayms(1000);
  74. initrgb(0,60,0,0,50);        //測試藍
  75. display();
  76. delayms(1000);
  77. initrgb(0,60,0,0,0);        //清屏
  78. display();
  79. delayms(1000);       
  80. EA = 1;   //開啟中斷,可接收數據
  81.        
  82.         while(1)
  83.                 {
  84.                 if(EA==0)//當中斷關閉,表示一幀數據接收完畢
  85.                         {
  86.                                 display();//將當前數據刷新
  87.                                 EA=1;//開啟中斷,接收下一幀數據
  88.                         }
  89.                 }       
  90. }

  91. void Uart() interrupt 4   //串口中斷
  92. {
  93.         uchar dat=0;
  94.     if (RI)
  95.     {
  96.                 dat=SBUF;
  97.         RI = 0;                 //清除RI位
  98.         if(dat==0x41&&in==0)
  99.                 {
  100.                         in=1;                                        //判斷數據0位(A)
  101.                         SBUF=in;
  102.                         return;
  103.                 }
  104.                 if(dat==0x64&&in==1)                //判斷數據1位(d)
  105.                 {
  106.                         in=2;
  107.                         SBUF=in;
  108.                         return;
  109.                 }
  110.                 else if(dat==0x61&&in==2)        //判斷數據2位(a)
  111.                 {
  112.                         in=3;
  113.                         SBUF=in;
  114.                         return;
  115.                 }
  116.                 else if(in==3)                                //讀取高位LED個數
  117.                 {
  118.                         h=dat;
  119.                         in++;
  120.                         return;
  121.                 }
  122.                 else if(in==4)                                //讀取低位LED個數
  123.                 {
  124.                         l=dat;
  125.                         in++;
  126.                         return;
  127.                 }
  128.                 else if(in==5)                                //讀取校驗值
  129.                 {
  130.                         o=dat;
  131.                         if(o!=(h^l^0x55)) in=0;                //校驗
  132.                         else in++;
  133.                         return;
  134.                 }
  135.                 else if(in>=6)                                //進入數據讀取階段
  136.                 {
  137.                         rgb[(in-6)/3][(in-6)%3]=dat;//將數據寫入數組
  138.                         if((in-6)/3==l)                                //根據發來的led個數確定數據是否接收完畢
  139.                         {
  140.                                 in=0;
  141.                                 EA=0;
  142.                                 return;
  143.                         }
  144.                         else in++;
  145.                         return;
  146.                 }
  147.                         else                          //如果in小于6,且中途數據中斷。則重新開始。
  148.                         {
  149.                                 in=0;
  150.                                 return;
  151.                         }
  152.         }
  153. }
復制代碼

所有資料51hei提供下載:
流光溢彩.rar (35.49 KB, 下載次數: 38)

評分

參與人數 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏7 分享淘帖 頂1 踩
回復

使用道具 舉報

沙發
ID:687694 發表于 2020-10-19 20:21 | 只看該作者
如果大家都想做一個玩玩,我就再出個詳細的教程帖子。畫好電路圖,PCB。
回復

使用道具 舉報

板凳
ID:820198 發表于 2021-4-12 10:52 | 只看該作者
不錯啊,坐等詳細教程
回復

使用道具 舉報

地板
ID:706724 發表于 2021-4-19 11:38 | 只看該作者
新手請教,程序中好像沒有把中斷的程序寫出來?IN0=0;EA=1;.......直接實現用EA=1/0來實現總中斷?
回復

使用道具 舉報

5#
ID:851858 發表于 2021-10-12 00:48 | 只看該作者
用89C52單片機燒上代碼,P1.0接燈帶數據端,不亮。
是89C52芯片不行,要stc15芯片? 還是要改代碼的哪里?
換過Keil軟件的器件為89C52,頭文件試過stc15w.h和reg52.h。
不太懂望指教。
arduino點亮過
回復

使用道具 舉報

6#
ID:851858 發表于 2021-10-12 01:00 | 只看該作者
本帖最后由 ddwy43 于 2021-10-12 01:03 編輯

arduino的點亮過,燈帶也就正常了
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 精品日韩一区二区 | 青青av | 国产一级在线 | 久久久久久99精品久久久 | 一区二区福利视频 | 欧美黄色网 | 91视频日本 | 国产精品一二三四 | 五月开心激情网 | 手机福利视频 | 超碰人人人 | 欧美成人一区二区三区片免费 | 国产精品亚洲精品 | 在线看日韩 | www.av在线视频 | 午夜视频在线播放 | 欧美视频a | 97在线免费| 国产免费无遮挡 | 精品一区二区在线播放 | 久久99精品久久久久久水蜜桃 | aaa级片| 欧美精品在线观看 | 福利色导航 | 欧美又大粗又爽又黄大片视频 | av小说在线观看 | 大桥未久在线视频 | 91亚洲精品乱码久久久久久蜜桃 | 欧美性生活网站 | 亚洲特级片 | 免费成人小视频 | 亚洲一区免费 | 国产日韩欧美 | 色婷婷丁香| 国产精品一区二区免费 | 网站av | 成人午夜在线 | 久久综合激情 | 欧美三级三级三级爽爽爽 | 免费99精品国产自在在线 | 免费看毛片的网站 |