0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

基于HT1621B段式液晶模塊的驅(qū)動應(yīng)用設(shè)計

電子設(shè)計 ? 來源:網(wǎng)絡(luò)整理 ? 作者:工程師1 ? 2018-05-25 03:48 ? 次閱讀

段式液晶由于其功耗低、價格便宜在很多家電中得到廣泛的應(yīng)用,其驅(qū)動其實并不復(fù)雜,大多是情況下都是用HT1621B進行驅(qū)動。

HT1621是128 點內(nèi)存映象和多功能的LCD驅(qū)動器HT1621 的軟件配置特性使它適用于多種LCD應(yīng)用場合包括LCD模塊和顯示子系統(tǒng)用于連接主控制器和HT1621的管腳只有4 或5 條HT1621 還有一個節(jié)電命令用于降低系統(tǒng)功耗。

在使用HT1621進行驅(qū)動時,首先得根據(jù)訂做的液晶進行原理設(shè)置。驅(qū)動液晶實際上就是往HT1621的內(nèi)部寄存器中寫數(shù)據(jù),至于數(shù)據(jù)如何去驅(qū)動液晶我們可以不去理會它。下面也一款訂做的液晶為例進行說明:

資源分配如下,3個數(shù)碼管每個數(shù)碼管由7段組成,還有3個風(fēng)速圖標(biāo),4個溫度圖標(biāo)和一個冒號圖標(biāo)。

我們知道HT1621是由4個COM口和18個Seg接口構(gòu)成,COM口的連接和簡單,直接對應(yīng)連接即可,而Seg可以根據(jù)你的PCB布局、連線的方便等進行選擇性連接。

在這里我們可以COM口對應(yīng)連接,Seg端口按照順序連接5~12腳,得到的圖紙如下:

144924V43_0.png

有了這個原理圖,后面我們就可以設(shè)計驅(qū)動程序了,在設(shè)計驅(qū)動程序之前,必須認(rèn)識到一個問題,段式液晶是由很多段或者圖標(biāo)、點構(gòu)成,從而構(gòu)成的顯示圖 案。而這些多、圖標(biāo)、點都是由HT1621的寄存器中的位組成的,所以,如果驅(qū)動程序按照位進行控制,將給我很大的方便和靈活。

但是我們知道,除了C51提供位操作為,其他單片機并不提供位操作的定義方式,但是,基本上所有的編譯器都提供位段的定義方式,所以下面我們將使用位段進行定義:

由原理圖和液晶資料我們可以看出,Seg0對應(yīng)第一個數(shù)碼管的F、G、E三段,Seg1對應(yīng)第一個數(shù)碼管的A、B、C、D四段。而第二個數(shù)碼管和第三個數(shù)碼 管的每一段順序與第一個相同。所以,我們可以使用與第一個數(shù)碼管相同的結(jié)構(gòu)體進行三個數(shù)碼管的定義,當(dāng)然有時候每個數(shù)碼管的每一段順序并不一定相同,這個 是由段式液晶在設(shè)計時的走線確定的。如果每一個數(shù)碼的順序不同,我們就得分別定義其結(jié)構(gòu)體了。

typedef union

{

struct

{

u8 DA : 1; //

u8 DB : 1; //

u8 DC : 1; //

u8 DD : 1; //

u8 Rcv : 4; //

} BtL;

struct

{ //

u8 DF : 1; //

u8 DG : 1;

u8 DE : 1; //

u8 DO : 1; //

u8 Rcv : 4; //

} BtH;

} HTB_SEG;

在這里,我們把同一個數(shù)碼管的7段定義在一個結(jié)構(gòu)體中,如果使用F、G、E三個段式,我們使用BtH這個變量,如果使用A、B、C、D四段時,我們使用 BtL這個變量。當(dāng)然,我們也可以把這兩個分開定義。由于第二個數(shù)碼管多了個冒號,同樣把其放入BtH變量中,第一個和第三個數(shù)碼管中沒有使用這個位,不 用即可。

typedef union

{

struct

{

u8 K1 : 1; //

u8 K2 : 1; //

u8 K3 : 1; //

u8 Rcv : 5; //

} BtL;

struct

{

u8 K7 : 1; //

u8 K6 : 1; //

u8 K5 : 1; //

u8 K4 : 1; //

u8 Rcv : 4; //

} BtH;

} HTB_ICN;

用同樣的方法定義剩余的圖標(biāo),獲得上面的結(jié)構(gòu)體。由此我們看出,每個寄存器實際上只使用了前面4個位,后面的4個位沒有使用,保留。

typedef struct

{

HTB_SEG Seg0;

HTB_SEG Seg1;

HTB_SEG Seg2;

HTB_SEG Seg3;

HTB_SEG Seg4;

HTB_SEG Seg5;

HTB_ICN Seg6;

HTB_ICN Seg7;

} HTB_RAM;

HTB_RAM HTBRam;

最后我們把使用的8個寄存器分別使用上面的結(jié)構(gòu)體變量進行定義,前面6個為數(shù)碼管,后面2個為圖標(biāo)。有了這個結(jié)構(gòu)體,后面定義一個變量用于操作每個數(shù)碼管。

數(shù)碼管顯示驅(qū)動如下,從0~9通過控制每一段形成字符:

/**************************************************************************************

* FunctionName : HTB_SegVal()

* Description : 數(shù)碼管填值

* EntryParameter : None

* ReturnValue : None

**************************************************************************************/

void HTB_SegVal(HTB_SEG *pSg1, HTB_SEG *pSg2, u8 dat)

{

switch (dat)

{

case 0: pSg2-》BtL.DA = 1; pSg2-》BtL.DB = 1; pSg2-》BtL.DC = 1; pSg2-》BtL.DD = 1;

pSg1-》BtH.DE = 1; pSg1-》BtH.DF = 1; pSg1-》BtH.DG = 0; break;

case 1: pSg2-》BtL.DA = 0; pSg2-》BtL.DB = 1; pSg2-》BtL.DC = 1; pSg2-》BtL.DD = 0;

pSg1-》BtH.DE = 0; pSg1-》BtH.DF = 0; pSg1-》BtH.DG = 0; break;

case 2: pSg2-》BtL.DA = 1; pSg2-》BtL.DB = 1; pSg2-》BtL.DC = 0; pSg2-》BtL.DD = 1;

pSg1-》BtH.DE = 1; pSg1-》BtH.DF = 0; pSg1-》BtH.DG = 1; break;

case 3: pSg2-》BtL.DA = 1; pSg2-》BtL.DB = 1; pSg2-》BtL.DC = 1; pSg2-》BtL.DD = 1;

pSg1-》BtH.DE = 0; pSg1-》BtH.DF = 0; pSg1-》BtH.DG = 1; break;

case 4: pSg2-》BtL.DA = 0; pSg2-》BtL.DB = 1; pSg2-》BtL.DC = 1; pSg2-》BtL.DD = 0;

pSg1-》BtH.DE = 0; pSg1-》BtH.DF = 1; pSg1-》BtH.DG = 1; break;

case 5: pSg2-》BtL.DA = 1; pSg2-》BtL.DB = 0; pSg2-》BtL.DC = 1; pSg2-》BtL.DD = 1;

pSg1-》BtH.DE = 0; pSg1-》BtH.DF = 1; pSg1-》BtH.DG = 1; break;

case 6: pSg2-》BtL.DA = 1; pSg2-》BtL.DB = 0; pSg2-》BtL.DC = 1; pSg2-》BtL.DD = 1;

pSg1-》BtH.DE = 1; pSg1-》BtH.DF = 1; pSg1-》BtH.DG = 1; break;

case 7: pSg2-》BtL.DA = 1; pSg2-》BtL.DB = 1; pSg2-》BtL.DC = 1; pSg2-》BtL.DD = 0;

pSg1-》BtH.DE = 0; pSg1-》BtH.DF = 0; pSg1-》BtH.DG = 0; break;

case 8: pSg2-》BtL.DA = 1; pSg2-》BtL.DB = 1; pSg2-》BtL.DC = 1; pSg2-》BtL.DD = 1;

pSg1-》BtH.DE = 1; pSg1-》BtH.DF = 1; pSg1-》BtH.DG = 1; break;

case 9: pSg2-》BtL.DA = 1; pSg2-》BtL.DB = 1; pSg2-》BtL.DC = 1; pSg2-》BtL.DD = 1;

pSg1-》BtH.DE = 0; pSg1-》BtH.DF = 1; pSg1-》BtH.DG = 1; break;

case 0: pSg2-》BtL.DA = 0; pSg2-》BtL.DB = 0; pSg2-》BtL.DC = 0; pSg2-》BtL.DD = 0;

pSg1-》BtH.DE = 0; pSg1-》BtH.DF = 0; pSg1-》BtH.DG = 0; break;

default:break;

}

}

/**************************************************************************************

* FunctionName : HTBColon()

* Description : 冒號

* EntryParameter : None

* ReturnValue : None

**************************************************************************************/

void HTBColon(OS_SWT swt)

{

HTBRam.Seg2.BtH.DO = (swt 》 0) ? 1 : 0;

}

/**************************************************************************************

* FunctionName : HTBTemStl()

* Description : 溫度

* EntryParameter : None

* ReturnValue : None

**************************************************************************************/

void HTBTemStl(u8 stl)

{

HTBRam.Seg7.BtH.K4 = 0;

HTBRam.Seg7.BtH.K5 = 0;

HTBRam.Seg7.BtH.K6 = 0;

HTBRam.Seg7.BtH.K7 = 0;

switch (stl)

{

case 0: HTBRam.Seg7.BtH.K4 = 1; break;

case 1: HTBRam.Seg7.BtH.K5 = 1; break;

case 2: HTBRam.Seg7.BtH.K6 = 1; break;

case 3: HTBRam.Seg7.BtH.K7 = 1; break;

default : break;

}

}

/**************************************************************************************

* FunctionName : HTBWndStl()

* Description : 風(fēng)速

* EntryParameter : None

* ReturnValue : None

**************************************************************************************/

void HTBWndStl(u8 stl)

{

HTBRam.Seg6.BtL.K1 = 0;

HTBRam.Seg6.BtL.K2 = 0;

HTBRam.Seg6.BtL.K3 = 0;

switch (stl)

{

case 0: HTBRam.Seg6.BtL.K3 = 1; break;

case 1: HTBRam.Seg6.BtL.K2 = 1; break;

case 2: HTBRam.Seg6.BtL.K1 = 1; break;

default : break;

}

}

圖標(biāo)的驅(qū)動如上,其實就是根據(jù)需要修改每一個寄存器位,這個寄存器修改后,我們還必須得傳遞給HT1621更新顯示,才能真正實現(xiàn)顯示的驅(qū)動:

/**************************************************************************************

* FunctionName : HTB_SendBitMsb()

* Description : 發(fā)送發(fā)送多位[高位在前]

* EntryParameter : None

* ReturnValue : None

**************************************************************************************/

void HTB_SendBitMsb(u8 dat, u8 cnt)

{

for (u8 i=0; i {

(dat & 0x80) ? GPIO_WriteHigh(HTB_DT_PORT, HTB_DT_PIN) :

GPIO_WriteLow(HTB_DT_PORT, HTB_DT_PIN);

dat 《《= 1;

GPIO_WriteLow(HTB_WR_PORT, HTB_WR_PIN);

HTB_DelayUs(3);

GPIO_WriteHigh(HTB_WR_PORT, HTB_WR_PIN);

}

}

/**************************************************************************************

* FunctionName : HTB_SendBitLsb()

* Description : 發(fā)送多位[低位在前]

* EntryParameter : None

* ReturnValue : None

**************************************************************************************/

void HTB_SendBitLsb(u8 dat, u8 cnt)

{

for (u8 i=0; i {

(dat & 0x01) ? GPIO_WriteHigh(HTB_DT_PORT, HTB_DT_PIN) :

GPIO_WriteLow(HTB_DT_PORT, HTB_DT_PIN);

dat 》》= 1;

GPIO_WriteLow(HTB_WR_PORT, HTB_WR_PIN);

HTB_DelayUs(3);

GPIO_WriteHigh(HTB_WR_PORT, HTB_WR_PIN);

}

}

/**************************************************************************************

* FunctionName : HTB_SendCmd()

* Description : 發(fā)送命令

* EntryParameter : None

* ReturnValue : None

**************************************************************************************/

void HTB_SendCmd(u8 cmd)

{

GPIO_WriteLow(HTB_CS_PORT, HTB_CS_PIN);

HTB_SendBitMsb(0x80, 3); // 前面3位命令代碼

HTB_SendBitMsb(cmd, 9); // 后面10位: a5~a0[RAM地址]+d3~d0[RAM數(shù)據(jù)]

GPIO_WriteHigh(HTB_CS_PORT, HTB_CS_PIN);

}

/**************************************************************************************

* FunctionName : HTBSendNDat()

* Description : 發(fā)送N數(shù)據(jù)

* EntryParameter : None

* ReturnValue : None

**************************************************************************************/

void HTBSendNDat(u8 addr, u8 *pDat, u8 cnt, u8 bitNum)

{

GPIO_WriteLow(HTB_CS_PORT, HTB_CS_PIN);

HTB_SendBitMsb(0xA0, 3); // 前面3位命令代碼

HTB_SendBitMsb(addr《《2, 6); // a5~a0[RAM地址]

for (u8 i=0; i {

HTB_SendBitLsb(*pDat++, bitNum); // RAM數(shù)據(jù)

}

GPIO_WriteHigh(HTB_CS_PORT, HTB_CS_PIN);

}

上面的函數(shù)是通過按位傳遞的方式把數(shù)據(jù)發(fā)給HT1621,并不復(fù)雜,這里就不相信介紹了:

typedef enum

{

HTB_CMD_BIAS = 0x29, // 0B:0010 abXc -ab控制占空比,-c控制偏壓

HTB_CMD_SYSEN = 0x01, //

HTB_CMD_LCDOFF = 0x02, //

HTB_CMD_LCDON = 0x03, //

} HTB_CMD;

最后,我們可以看出,在修改了全局變量后,在把更新的數(shù)據(jù)傳遞給驅(qū)動芯片就可以了,非常簡單方便靈活,這個示例讓我們充分了解和使用位段進行位控制是非常方便。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 單片機
    +關(guān)注

    關(guān)注

    6037

    文章

    44561

    瀏覽量

    635655
  • LCD驅(qū)動器
    +關(guān)注

    關(guān)注

    1

    文章

    59

    瀏覽量

    12717
收藏 人收藏

    評論

    相關(guān)推薦

    點陣液晶控制器燃氣表段碼驅(qū)動芯片VK1621S

    控制器和驅(qū)動器系列: VK1024B 2.4V~5.2V 6seg4com 63 6*2 偏置電壓1/2 1/3 S0P-16 VK1056B 2.4V~5.2V 14seg4com 143 14*2
    發(fā)表于 12-18 17:25

    電量顯示液晶驅(qū)動小體積液晶驅(qū)動芯片VK1088B

    產(chǎn)品型號:VK1088B 產(chǎn)品品牌:永嘉微電/VINKA 封裝形式:QFN32 (4mm×4mm) 產(chǎn)品年份:新年份 VK1088B概述: VK1088B 是一個22*4的LCD驅(qū)動
    發(fā)表于 12-09 17:27

    基于FPGA的LCD1602液晶顯示模塊驅(qū)動設(shè)計

    本文通過以LCD1602液晶顯示模塊為基礎(chǔ),介紹FPGA驅(qū)動LCD1602原理,詳細介紹硬件原理圖設(shè)計及FPGA驅(qū)動LCD1602軟件設(shè)計,通過萬年歷功能綜合實現(xiàn)時鐘功能、LCD160
    的頭像 發(fā)表于 10-24 14:42 ?1342次閱讀
    基于FPGA的LCD1602<b class='flag-5'>液晶</b>顯示<b class='flag-5'>模塊</b><b class='flag-5'>驅(qū)動</b>設(shè)計

    常見液晶驅(qū)動控制芯片詳解

    常見液晶驅(qū)動控制芯片詳解 前言 因此各位朋友在選擇 LCD液晶模塊的時候,在考慮到串行,還是并行的方式時,可根據(jù)其驅(qū)動控制 IC的型號來判別,當(dāng)然你還需要看你選擇的LCD
    的頭像 發(fā)表于 10-12 16:35 ?1537次閱讀

    通用抗干擾段碼屏驅(qū)動液晶驅(qū)動芯片LCD驅(qū)動芯片VK1626(48SEGx16COM)

    顯示IC、液晶驅(qū)動IC、液晶驅(qū)動芯片、LCD芯片、液晶芯片、液晶
    發(fā)表于 10-12 15:44

    超低功耗LCD段筆式液晶顯示驅(qū)動芯片VKL092Q簡介

    、液晶IC、段碼驅(qū)動顯示IC、筆段式液晶驅(qū)動、LCD液晶顯示
    發(fā)表于 07-15 15:51

    高抗干擾段碼LCD液晶顯示屏驅(qū)動控制電路(芯片)-VK2C22A/B

    一款超強抗干擾的LCD液晶段碼屏顯示驅(qū)動控制芯片-VK2C22A/B LQFP52/48 可驅(qū)動44*4點的LCD顯示屏,ESD達8KV,完美兼容替代市面16C22 概述:S88+57
    發(fā)表于 06-22 14:57

    COG封裝的高EMC抗擾度段式LCD液晶驅(qū)動CN91C4SXX系列

    COG封裝的高EMC抗擾度段式LCD液晶驅(qū)動CN91C4SXX系列
    的頭像 發(fā)表于 06-06 09:54 ?825次閱讀
    COG封裝的高EMC抗擾度<b class='flag-5'>段式</b>LCD<b class='flag-5'>液晶</b><b class='flag-5'>驅(qū)動</b>CN91C4SXX系列

    低功耗、高EMC抗擾度段式LCD液晶驅(qū)動芯片CN9101C4S32

    低功耗、高EMC抗擾度段式LCD液晶驅(qū)動芯片CN9101C4S32
    的頭像 發(fā)表于 06-03 10:10 ?631次閱讀
    低功耗、高EMC抗擾度<b class='flag-5'>段式</b>LCD<b class='flag-5'>液晶</b><b class='flag-5'>驅(qū)動</b>芯片CN9101C4S32

    低功耗高EMC抗擾度段式LCD液晶驅(qū)動芯片CN9001C4S36

    低功耗高EMC抗擾度段式LCD液晶驅(qū)動芯片CN9001C4S36
    的頭像 發(fā)表于 05-28 10:02 ?634次閱讀
    低功耗高EMC抗擾度<b class='flag-5'>段式</b>LCD<b class='flag-5'>液晶</b><b class='flag-5'>驅(qū)動</b>芯片CN9001C4S36

    CMOS液晶顯示驅(qū)動器CD4054B CD4055B CD4056B TYPES數(shù)據(jù)表

    電子發(fā)燒友網(wǎng)站提供《CMOS液晶顯示驅(qū)動器CD4054B CD4055B CD4056B TYPES數(shù)據(jù)表.pdf》資料免費下載
    發(fā)表于 05-20 09:35 ?0次下載
    CMOS<b class='flag-5'>液晶</b>顯示<b class='flag-5'>驅(qū)動</b>器CD4054<b class='flag-5'>B</b> CD4055<b class='flag-5'>B</b> CD4056<b class='flag-5'>B</b> TYPES數(shù)據(jù)表

    STM8L152使用寄存器操作過段式液晶,初始化之后不能用是為什么?怎么解決?

    哪位大神使用寄存器操作過段式液晶,可以指導(dǎo)一下嗎。我初始化之后還是不能用。
    發(fā)表于 05-09 08:06

    液晶模塊是什么?液晶模塊液晶顯示屏有什么區(qū)別?

    在電子顯示技術(shù)日益發(fā)展的今天,液晶顯示技術(shù)以其獨特的優(yōu)勢占據(jù)了市場的主導(dǎo)地位。液晶模塊液晶顯示屏作為液晶顯示技術(shù)的兩大核心組件,雖然聯(lián)系緊密,但在功能和結(jié)構(gòu)上卻存在著明顯的區(qū)別。跟著
    的頭像 發(fā)表于 03-29 09:45 ?1280次閱讀

    PC1621K LCD 驅(qū)動電路數(shù)據(jù)手冊

    電子發(fā)燒友網(wǎng)站提供《PC1621K LCD 驅(qū)動電路數(shù)據(jù)手冊》資料免費下載
    發(fā)表于 03-12 11:01 ?0次下載

    字符液晶模塊灰度顯示

    正如STN液晶面板試玩中所解釋的,STN液晶顯示器的灰度顯示是通過逐幀的點ON/OFF比例控制(所謂的PWM)來進行的。由于STN液晶的反應(yīng)較慢,ON/OFF以平均濃度顯示。圖形 LCD 控制器在硬件中處理此問題(三維抖動)。
    發(fā)表于 02-02 16:54 ?418次閱讀
    字符<b class='flag-5'>液晶模塊</b>灰度顯示