什么? STC8H4K32TL-40MHz-LQFP48/32, QFN48/32
16個觸摸按鍵,硬件自動刷新驅動32個8段LED數(shù)碼管,或16個米字形數(shù)碼管
80mA大電流硬件LED數(shù)碼管自動刷新驅動器居然可以8位亮度256種組合?
怎么實現(xiàn)的!!!1.6 !
1111.png (1.27 MB, 下載次數(shù): 13)
下載附件
2024-6-5 09:10 上傳
222.png (394.57 KB, 下載次數(shù): 11)
下載附件
2024-6-5 09:10 上傳
因為我要用電位器調米字數(shù)碼管的亮度,但是STC8H4K64TL只支持3位亮度,調起來一級一級的蹦看起來很明顯,一點也不絲滑,想了很多辦法都不能很優(yōu)雅的解決這個問題,后來過了幾天靈機一動就解決了,在此我把這個方法命名為STC8H4K64TL的“硬3軟5”80mA大電流硬件LED驅動器亮度調節(jié)方法,不廢話了,具體看程序,無關代碼已刪除,關鍵代碼都寫了備注。
程序:
unsigned char SEG_Bri; //8bit亮度,范圍0~251
void PWM_Func(void)
{
SEG_Bri=(unsigned char)((pow((float)Filter_Value,2.2F))/351204.0F); //Gamma校正,輸入Filter_Value,范圍0~4095,輸出SEG_Bri,范圍0~251
}
void SEG_Func(unsigned char temp) //輸入temp,范圍0~8,對應亮度0%、12.5%、25%、37.5%、50%、62.5%、75%、87.5%、100%
{
if(temp>8) //無效數(shù)據(jù)限幅
{
temp=8;
}
if(temp) //大于0亮
{
LEDCTRL=7-(temp-1)|0x90; //計算LEDCTRL
COMEN=0x1F; //開COM
SEGENL=0xFF; //開SEG
SEGENH=0x3F; //開SEG
}
else //否則滅
{
COMEN=0x00; //關COM
SEGENL=0x00; //關SEG(不關低亮度有鬼影,僅在硬件亮度頻繁在0和1之間切換時有)
SEGENH=0x00; //關SEG(不關低亮度有鬼影,僅在硬件亮度頻繁在0和1之間切換時有)
LEDCTRL=0x97; //無法關閉,只能設置為最低亮度
}
COM0_DA_L=(unsigned char)SEG_Buf[SEG_Addr];
COM0_DA_H=(unsigned char)(SEG_Buf[SEG_Addr]>>8)&0x3F;
COM1_DA_L=(unsigned char)SEG_Buf[SEG_Addr+1];
COM1_DA_H=(unsigned char)(SEG_Buf[SEG_Addr+1]>>8)&0x3F;
COM2_DA_L=(unsigned char)SEG_Buf[SEG_Addr+2];
COM2_DA_H=(unsigned char)(SEG_Buf[SEG_Addr+2]>>8)&0x3F;
COM3_DA_L=(unsigned char)SEG_Buf[SEG_Addr+3];
COM3_DA_H=(unsigned char)(SEG_Buf[SEG_Addr+3]>>8)&0x3F;
COM4_DA_L=(unsigned char)(SEG_Buf[SEG_Addr]>>15)&0x01;
COM4_DA_H=0x00;
}
void Init(void)
{
P_SW2|=EAXFR;
//PxMx在這里設置一下,硬件LED驅動器相關IO設置成推挽
P2DR=0x00;
AUXR=0x55; //設置定時器0為12T模式,設置定時器1為1T模式,使能定時器2,設置定時器2為1T模式,選擇定時器2作為波特率發(fā)生器
TMOD=0x01; //設置定時器0為16位不自動重載模式,設置定時器1為16位自動重載模式
TM2PS=0x00; //設置定時器2分頻器
TL1=(0x10000-6400); //設置定時器1初始值(6400)頻率同步,160*8*5=6400
TH1=(0x10000-6400)>>8; //設置定時器1初始值(6400)頻率同步,160*8*5=6400
TF1=0; //清除TF1中斷標志位
TR1=1; //打開定時器1
ET1=1; //啟用定時器1中斷
COMEN=0x1F; //使用的COM
SEGENL=0xFF; //使用的SEG
SEGENH=0x3F; //使用的SEG
LEDCTRL=0x90; //默認最大亮度
LEDCKS=0x00; //最高頻率不分頻
EA=1;
COM0_DA_L=0x00;
COM0_DA_H=0x00;
COM1_DA_L=0x00;
COM1_DA_H=0x00;
COM2_DA_L=0x00;
COM2_DA_H=0x00;
COM3_DA_L=0x00;
COM3_DA_H=0x00;
COM4_DA_L=0x00;
COM4_DA_H=0x00;
}
void main(void)
{
Init();
while(1)
{
PWM_Func();
}
}
void Timer1_Isr(void) interrupt 3
{
static unsigned char temp,temph,templ;
if(SEG_Bri>0&&SEG_Bri<=4) //1、2、3都按4處理,否則極低亮度閃爍明顯,因為硬件LED驅動器運行到哪里不知道,也沒有中斷,頻率不能絕對同步
{
SEG_Bri=4;
}
temph=SEG_Bri/28; //因為硬件量化級數(shù)是0~8共9級,9的倍數(shù)在8位之內最大值為252,所以SEG_Bri為0~251,軟件量化級數(shù)為252/9=28,temph為MSB,給硬件,范圍0~8
templ=SEG_Bri%28; //因為硬件量化級數(shù)是0~8共9級,9的倍數(shù)在8位之內最大值為252,所以SEG_Bri為0~251,軟件量化級數(shù)為252/9=28,templ為LSB,給軟件,范圍0~27
if(templ<=temp) //和模擬電路PWM類似,temp為上升鋸齒波,templ為固定值,二者比較即可形成PWM,如果等于也按本級亮度處理,不然最高亮度temph溢出
{
SEG_Func(temph); //硬件為本級亮度
}
else
{
SEG_Func(temph+1); //硬件為下一級亮度
}
temp++; //計數(shù)器掃描,范圍0~27
if(temp==28) //溢出
{
temp=0; //清零
}
}
|