當前位置:首頁 > 嵌入式 > 嵌入式硬件

  引言

  環(huán)境光傳感器(ALS)集成電路正越來越多地用于各種顯示器和照明設備,以節(jié)省電能,改善用戶體驗。借助ALS解決方案,系統(tǒng)設計師可根據環(huán)境光強度,自動調節(jié)顯示屏的亮度。因為背光照明的耗電量在系統(tǒng)的總耗電量中占據很大的比例,實行動態(tài)的背光亮度控制,可節(jié)省大量的電能。此外,它還能夠改善用戶體驗,讓顯示屏亮度根據環(huán)境光條件自行調整到最佳狀態(tài)。

  系統(tǒng)實現(xiàn)需要三大部分:監(jiān)測環(huán)境光強的光傳感器、數據處理裝置(通常是微控制器)、控制背光輸入電流的執(zhí)行器。

  背光控制:環(huán)境光傳感器

  圖1是實施背光控制的系統(tǒng)示范框圖。在這套組合中,光傳感器是關鍵的組成部分,因為它要向系統(tǒng)的其他模塊提供環(huán)境光強信息。光傳感器必須具備將光信號轉換成電信號的信號轉換器(譬如光電二極管或CdS光敏電阻)和信號放大和/或調節(jié)裝置以及模/數轉換器(ADC)。

圖1. 實施背光控制的系統(tǒng)框圖

  圖2所示為分立光電二極管電路,從圖中可以看出,該電路需要一個或多個運算放大器:一個用于電流到電壓的轉換,可能還需要一級放大,提供附加增益。它還包括一些分支電路,用于供電,確保高度可靠的信號鏈。而在空間極其寶貴的應用中,所需元件的數量過多可能導致空間受限問題。

圖2. 光電二極管電路分立設計

  這里還存在一個更細微的問題。具體而言,理想情況下,應確保環(huán)境光的測量模擬了人眼對光線的響應機制。這通常借助CIE提供的視覺亮度曲線(圖3)。然而,光電二極管很少能夠完全模擬這種響應機制,因為它們通常具有很高的紅外(IR)靈敏度。在IR強度較大的光照條件(譬如白熾燈或日光)下,這種紅外靈敏度會造成錯誤地判斷光線強度。

  解決上述問題的方法之一是使用兩個光電二極管:一個采用對可見光和紅外光都很敏感的元件,另一個采用只對紅外光敏感的元件。最終用前者的響應值減去后者的響應值,將紅外干擾降至最小,獲得準確的可見光響應。

  這種解決方案雖然有效,卻增加了分立電路的占用空間。通常還很難、甚至不可能讓兩個分立的光電二極管配合得足夠緊密,以實現(xiàn)消除紅外干擾的目的。如果不配備精密放大器(譬如對數放大器),動態(tài)范圍可能很小。換句話說,很難利用這種組合獲得可重復的結果。

圖3. CIE曲線和典型的光電二極管

  高集成度解決方案不僅能夠獲得比人眼光學系統(tǒng)更真實的光強數據,還能夠節(jié)省大量空間。MAX9635、MAX44009等環(huán)境光傳感器,可將所有信號調節(jié)和模/數轉換器集成在一個小封裝(2mm x 2mm UTDFN封裝)內,從而在空間受限應用中有效節(jié)省電路板面積。

  圖4提供了MAX44009的功能框圖,采用I?C通信協(xié)議,使其與微控制器的連接方式更簡單,數據傳輸速度更快。除此之外,該解決方案的高集成特性使其能夠置于柔性電纜,安裝在離主電路板距離合適的位置。

圖4. MAX44009功能框圖

  背光控制:調節(jié)顯示屏亮度

  該控制方案的第二部分是調節(jié)顯示屏的背光亮度。這可通過多種方式實現(xiàn),具體取決于設備中的顯示屏模塊。有兩種最簡單的方式,一種是借助脈沖寬度調制(PWM)方案的直接調節(jié)方式,另一種是采用顯示屏控制器的間接調節(jié)方式。

  許多顯示屏模塊如今都配有一個集成控制器,用戶可以通過向控制器發(fā)送串行命令,直接設置背光亮度。如果顯示屏模塊未配備集成控制器,還可安裝一個簡單的背光控制執(zhí)行器,控制顯示屏后面用于背光照明的白光LED燈的輸入電流。實現(xiàn)這種控制的一種簡單辦法是:直接給LED串聯(lián)一個場效應晶體管(FET),使用PWM信號快速打開、關閉FET (圖5)。然而,也可以利用單一芯片(用于LED控制的MAX1698升壓轉換器)準確、可靠地調節(jié)(圖6),請參考應用筆記3866“Low-power PWM output controls LED brightness”,獲取詳細信息。

圖5. 簡單的PMW控制電路

圖6. 基于MAX1698的LED亮度調節(jié)器

  背光控制:建立連接

  最后一步就是在傳感器和執(zhí)行器之間建立連接,通過微控制器實現(xiàn)。有人可能首先要問:“環(huán)境光強如何映射到背光亮度?”事實上,有些文獻專門介紹了相關方案。其中一種映射方式是,Microsoft?針對運行Windows? 7?操作系統(tǒng)的計算機提出的。圖7所示曲線是由Microsoft提供的,它可以將環(huán)境光強度映射到顯示屏亮度(以全部亮度的百分比表示)。

圖7. 將環(huán)境光強映射為最佳顯示屏亮度的曲線示例

  這種特殊曲線可以用以下函數表示:

  如果設備采用的是已集成亮度控制功能的LCD控制芯片,就可通過向芯片發(fā)送指令,輕松設置背光亮度。如果設備采用的是PWM直接控制亮度,則要考慮如何將比例信號映射至顯示屏亮度。

  在MAX1698示例中,根據其產品說明書的介紹,可以將驅動電流映射為電壓。通過這個示例,我們可以假設LED電流強度幾乎與其電流呈線性關系。這樣,我們就可以在上述等式中乘上一個系數,計算出PWM所映射的有效電壓,該電壓再被映射至LED電流,最后轉化成顯示屏亮度。

  方案實施

  最好不要從一個亮度級直接跳轉到另一個亮度級,而是平滑上調和下調背光亮度,確保不同亮度等級之間無縫過渡。為了達到這一目的,最好采用帶有固定或不同亮度步長、可逐步調節(jié)亮度的定時中斷,也可采用帶有可控制LED輸入電流的PWM值的定時中斷,或者是能夠發(fā)送到顯示屏控制器的串行指令的定時中斷。圖8提供了這種算法的一個示例。

圖8. 步進式亮度調節(jié)的算法示例

  另一個問題是,系統(tǒng)響應環(huán)境光強變化的速度。我們應盡量避免過快地改變亮度等級。這是因為光強的瞬間變化(譬如一扇窗戶打開或瞬間有一束光掃過)可能導致背光亮度發(fā)生不必要的變化,這往往會造成用戶感覺不適。并且,較長的響應時間還有助于減少微控制器對光傳感器的檢測次數,從而可以釋放一定的微控制器資源。

  最初級的方法就是每隔一兩秒鐘檢查一次光傳感器,然后相應地調整背光亮度。更好的方法是,只有光線強度偏離特定范圍一定時間后,才對背光亮度進行調節(jié)。譬如,如果正常光強是200lux,我們可能只會在光強降到180lux以下或升至220lux以上,而且持續(xù)時間超過數秒的情況下才調節(jié)亮度。幸運的是,MAX9635和MAX44009都集成了中斷引腳和閾值寄存器,可輕松實現(xiàn)這個目的。請參考應用筆記4786“MAX9635環(huán)境光傳感器的接口程序”,獲取更多詳細信息。

  附錄:源代碼

#define MAX44009_ADDR0x96// begin definition of slave addresses for MAX44009#define INT_STATUS0x00#define INT_ENABLE0.01#define CONFIG_REG0x02#define high_BYTE0x03#define LOW_BYTE0x04#define THRESH_HIGH0x05#define THRESH_LOW0x06#define THRESH_TIMER0x07// end definition of slave addresses for MAX44009 extern float SCALE_FACTOR;// captures scaling factors to map from % brightness to PWMfloat currentBright_pct;// the current screen brightness, in % of maximumfloat desiredBright_pct;// the desired screen brightness, in % of maximumfloat stepSize;// the step size to use to go from the current // brightness to the desired brightnessuint8 lightReadingCounter; /** *Function:SetPWMDutyCycle * *Arguments:uint16 dc - desired duty cycle * *Returns:none * *Description:Sets the duty cycle of a 16-bit PWM, assuming that in this  *architecture, 0x0000 = 0% duty cycle * 0x7FFF = 50% and 0xFFFF = 100%**/extern void SetPWMDutyCycle(uint16 dc); /** *Function:I2C_WriteByte * *Arguments:uint8 slaveAddr - address of the slave device *uint8 command - destination register in slave device *uint8 data - data to write to the register * *Returns:ACK bit * *Description:Performs necessary functions to send one byte of data to a  *specified register in a specific device on the I2C bus**/uint8 2C_WriteByte(uint8 slaveAddr, uint8 command, uint8 data); /** *Function:I2C_ReadByte * * Arguments:uint8 slaveAddr - address of the slave device *uint8 command - destination register in slave device *uint8 *data - pointer data to read from the register *  *Returns:ACK bit * *Description:Performs necessary functions to get one byte of data from a  *specified register in a specific deviceon the I2C bus**/uint8 I2C_ReadByte(uint8 slaveAddr, uint8 command, uint8* data); /** *Function:getPctBrightFromLuxReading * *Arguments:float lux - the pre-computed ambient light level * *Returns:The % of maximum brightness to which the backlight should be set  *given the ambient light (0 to 1.0) * *Description:Uses a function to map the ambient light level to a backlight  *brightness by using a predetermined function**/float getPctBrightFromLuxReading(float lux); /** *Function:mapPctBrighttoPWM * *Arguments:float pct * *Returns:PWM counts needed to achieve the specified % brightness (as  *determined by some scaling factors)**/uint16 mapPctBrighttoPWM(float pct); /** *Function:getLightLevel * *Arguments:n/a * *Returns:the ambient light level, in lux * *Description:Reads both the light registers on the device and returns the  *computed light level**/float getLightLevel(void); /** *Function:stepBrightness * *Arguments:n/a * *Returns:n/a * *Description:This function would be called by an interrupt. It looks at the  *current brightness setting, then the desired brightness setting.  *If there is a difference between the two, the current brightness  *setting is stepped closer to its goal.**/void stepBrightness(void); /** *Function:timerISR * *Arguments:n/a * *Returns:n/a * *Description:An interrupt service routine which fires every 100ms or so. This  *handles all the ambient light sensor and backlight *control code.**/void timerISR(void); void main() {SetupMicro();// some subroutine which initializes this CPUI2C_WriteByte(MAX44009_ADDR, CONFIG_REG, 0x80);// set to run continuouslylightReadingCounter = 0;stepSize = .01;currentBright_pct = 0.5;desiredBright_pct = 0.5;SetPWMDutyCycle(mapPctBrighttoPWM(currentBright_pct));InitializeTimerInterrupt();// set this to fire every 100mswhile(1) {// do whatever else you need here, the LCD control is done in interruptsIdle();}} // main routine // the point at which the function clips to 100%#define MAXIMUM_LUX_BREAKPOINT1254.0float getPctBrightFromLuxReading(float lux) {if (lux > MAXIMUM_LUX_BREAKPOINT)return 1.0;elsereturn (9.9323*log(x) + 27.059)/100.0;} // getPctBrightFromLuxReading uint16 mapPctBrighttoPWM(float pct) {return (uint16)(0xFFFF * pct * SCALE_FACTOR);} // mapPctBrighttoPWM float getLightLevel(void) {uint8* lowByte;uint8* highByte;uint8 exponent;uint8 mantissa;float result;I2C_ReadByte(MAX44009_ADDR, HIGH_BYTE, highByte);I2C_ReadByte(MAX44009_ADDR, LOW_BYTE, lowByte);exponent = (highByte & 0xF0) >> 4;// upper four bits of high byte registermantissa = (highByte & 0x0F) << 4;// lower four bits of high byte register =    // upper four bits of mantissamantissa += lowByte & 0x0F;   // lower four bits of low byte register =   // lower four bits of mantissaresult = mantissa * (1 << exponent) * 0.045;return result;} //getLightLevel void stepBrightness(void) { // if current is at desired, don't do anythingif (currentBright_pct == desiredBright_pct)return;// is the current brightness above the desired brightness?else if (currentBright_pct > desiredBright_pct) {// is the difference between the two less than one step?if ( (currentBright_pct-stepSize) < desiredBright_pct)currentBright_pct = desiredBright_pct;elsecurrentBright_pct -= stepSize;} // else ifelse if (currentBright_pct < desiredBright_pct) {// is the difference between the two less than one step?if ( (currentBright_pct+stepSize) > desiredBright_pct)currentBright_pct = desiredBright_pct;elsecurrentBright_pct += stepSize;} // else ifSetPWMDutyCycle(mapPctBrighttoPWM(currentBright_pct));return;} // stepBrightness void timerISR(void) {float lux;float pctDiff;stepBrightness();if (lightReadingCounter)lightReadingCounter--;else {lightReadingCounter = 20;// 2 second delaylux = getLightLevel();desiredBright_pct = getPctBrightFromLuxReading(lux);pctDiff = abs(desiredBright_pct - currentBright_pct);stepSize = (pctDiff <= 0.01) ? 0.01:pctDiff/10;} // elseClearInterruptFlag();} // timerISR


本站聲明: 本文章由作者或相關機構授權發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內容真實性等。需要轉載請聯(lián)系該專欄作者,如若文章內容侵犯您的權益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或將催生出更大的獨角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數字化轉型技術解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關鍵字: AWS AN BSP 數字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術公司SODA.Auto推出其旗艦產品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關鍵字: 汽車 人工智能 智能驅動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務中斷的風險,如企業(yè)系統(tǒng)復雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務連續(xù)性,提升韌性,成...

關鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據媒體報道,騰訊和網易近期正在縮減他們對日本游戲市場的投資。

關鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數據產業(yè)博覽會開幕式在貴陽舉行,華為董事、質量流程IT總裁陶景文發(fā)表了演講。

關鍵字: 華為 12nm EDA 半導體

8月28日消息,在2024中國國際大數據產業(yè)博覽會上,華為常務董事、華為云CEO張平安發(fā)表演講稱,數字世界的話語權最終是由生態(tài)的繁榮決定的。

關鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應對環(huán)境變化,經營業(yè)績穩(wěn)中有升 落實提質增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務引領增長 以科技創(chuàng)新為引領,提升企業(yè)核心競爭力 堅持高質量發(fā)展策略,塑強核心競爭優(yōu)勢...

關鍵字: 通信 BSP 電信運營商 數字經濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術學會聯(lián)合牽頭組建的NVI技術創(chuàng)新聯(lián)盟在BIRTV2024超高清全產業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術創(chuàng)新聯(lián)...

關鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關鍵字: BSP 信息技術
關閉
關閉