S3C2440芯片內部共有8路A/D轉換通道,AIN0-AIN7,轉換器只有一個,轉換精度為10位,最大轉換率為2.5MHzA/D轉換器時鐘下的500KSPS。A/D轉換器支持片上采樣-保持功能和掉電模式的操作。在常見的設計中,一般AIN4,AIN5,AIN6,AIN7被用作四線電阻觸摸屏的YM、YP、XM、XP通道,剩余的AIN0~AIN3被引出,其中AI0外接一個可調電阻。ADC的配置流程如下:
本文引用地址:http://m.butianyuan.cn/article/201611/321746.htm
1、ADCDLY(P446)
rADCDLY=50000;//Normalconversionmodedelayabout(1/3.6864M)*50000=13.56ms
2、ADCCON(P444)的設置,選擇轉換通道和設置轉換頻率
ADCCON[0],AD轉換開始有效,1轉換開始且該位在轉換開始后變?yōu)?,通過這個特點可以判斷是否開始轉換。什么意思呢,設為1只是認為地讓它轉換開始,但是未必開始轉換,還必須通過while循環(huán)查詢方式判斷該位是否變?yōu)?,變?yōu)?表示轉換開始。
ADCCON[1],AD轉換通過讀取有效,1通過讀取操作有效;ADCCON[2],備用操作模式選擇,0普通操作模式,1,備用操作模式;
ADCCON[5:3]:轉換通道選擇;
000=AIN0
001=AIN1
010=AIN2
011=AIN3
100=YM
101=YP
110=XM
111=XP
ADCCON[13:6],AD轉換器預分頻器值0~255,ADC頻率應該小于PCLK的1/5;
ADCCON[14],AD轉換器預分頻器使能,1使能;
ADCCON[15],AD轉換結束標志,0:轉化過程中,1:轉換結束;
ADC初始化程序示例如下:
#defineADC_FREQ2500000//希望的ADC轉換頻率
volatileU32preScaler;
volatileU32adc_value=0;//在程序開始處聲明
voidadc_init(void)
{
//選擇輸入通道,AIN0,對應開發(fā)板上W1可調電阻
intchannel=0;
preScaler=ADC_FREQ;
preScaler=50000000/ADC_FREQ-1;//PCLK=50M
rADCCON=(1<<14)"(preScaler<<6)|(channel<<3);//setupchannel
delay(1000);
}
3、ADCDAT0(ADCDAT1)P447,讀取轉換值
ADCDAT0[9:0],X坐標轉換結果值,包括普通模ADC轉換結果值;
(ADCDAT1[9:0],Y坐標轉換結果;)
ADC轉換程序示例,通過輪詢方式
voidMain(void)
{
Set_Clk();
adc_init();
while(1)
{
adc_value=ReadAdc(0);
delay(1000);
}
}
intReadAdc(intchannel)
{
rADCCON|=0x01;//startADC
while(rADCCON&0x1);//checkifEnable_startislow
while(!(rADCCON&0x8000));//checkifEC(EndofConversion)flagishigh
return((int)rADCDAT0&0x3ff);
}
ADC通過中斷方式讀取轉換值
注意ADC的中斷有兩個子中斷,INT_ADC_S和INT_TC需要先處理一下子中斷INT_ADC_S,再處理INT_ADC。
voidadc_init(void)
{
intchannel=0;/選擇輸入通道,AIN0,對應開發(fā)板上W1可調電阻
preScaler=ADC_FREQ;//設置分頻時鐘
preScaler=50000000/ADC_FREQ-1;//PCLK=50M
rADCCON=(1<<14)|(preScaler<<6)|(channel<<3);
ClearSubPending(BIT_SUB_ADC);//清子中斷處理寄存器ClearPending(BIT_ADC);//清中斷處理寄存器
pISR_ADC=(U32)adc_ISR;
EnableSubIrq(BIT_SUB_ADC);//開AD子中斷
EnableIrq(BIT_ADC);//開AD中斷
delay(1000);
}
void__irqadc_ISR(void)
{
intadc_value;//adc_value應該設為全局變量,這里放這里以便分析
ClearSubPending(BIT_SUB_ADC);//清子中斷處理寄存器ClearPending(BIT_ADC);//清中斷處理寄存器
adc_value=(int)rADCDAT0&0x3ff;
}
觸摸屏工作流程以及程序設計流程:
一、觸摸屏初始化:
1、ADCTSC設置(P445),
[1:0]:11設置觸摸屏接口為中斷等待模式,等待觸摸筆按下
[2]:0ADC普通轉換模式,1自動連續(xù)測量X坐標和Y坐標
[3]:0XP上拉有效,1XP上拉無效
2、ClearSubPending(BIT_SUB_ADC);//清子中斷處理寄存器ClearPending(BIT_ADC);//清中斷處理寄存器
ClearSubPending(BIT_SUB_TC);
清除源掛起寄存器(SRCPND)、中斷掛起寄存器(INTPND)、子源掛起寄存器(SUBSRCPND)。注意有兩個中斷,觸摸屏中斷:當觸摸筆按下或抬起產(chǎn)生的中斷,ADC中斷:觸摸屏坐標AD轉換結束產(chǎn)生的中斷。
3、EnableSubIrq(BIT_SUB_ADC);//開AD子中斷
EnableIrq(BIT_ADC);//開AD中斷
EnableSubIrq(BIT_SUB_TC);//開AD子中斷TC
關中斷屏蔽寄存器和子中斷屏蔽寄存器(INTMSK,INTSUBMSK)。4、pISR_ADC=(U32)AdcTsAuto;
程序入口函數(shù),中斷模式和中斷優(yōu)先級默認即可。
二、觸摸屏中斷服務子程序:
一)觸摸筆按下中斷
4、如果中斷發(fā)生,設置x,y坐標為自動轉換模式
rADCTSC=(1<<3)|(1<<2);
[2]:0ADC普通轉換模式,1自動連續(xù)測量X坐標和Y坐標
[3]:0XP上拉有效,1XP上拉無效
5、啟動AD轉換,然后檢測AD轉換是否啟動
rADCCON|=0x1;//startADC
while(rADCCON&0x1);//checkifEnable_startislow
6、檢測AD轉換是否結束,若結束,獲取x,y坐標的值
通過輪詢方式,也可以是中斷方式判斷轉換結束。
while(!(rADCCON&0x8000));//checkifEC(EndofConversion)flagishigh,Thislineisnecessary~!!
while(!(rSRCPND&0x80000000));//checkifADCisfinishedwithinterruptbit
xdata=(rADCDAT0&0x3ff);
ydata=(rADCDAT1&0x3ff);
7、對幾個寄存器寫1清零,防止反復發(fā)生中斷(這里的中斷是筆尖按下中斷)
ClearSubPending(BIT_SUB_TC);
ClearPending(BIT_ADC);
/rSRCPND=0x80000000;rINTPND=0x80000000;也可以
8、再次允許中斷
允許觸摸筆被彈起的中斷
EnableSubIrq(BIT_SUB_TC);
EnableIrq(BIT_ADC);//rINTMSK=0x7fffffff;
二)觸摸筆抬起中斷
9、設置觸摸屏即可為等待中斷模式,等待觸摸筆抬起(ADCTSC,關鍵是要設置觸摸筆抬起中斷信號)
rADCTSC=0xd3;//Waitingforinterrupt
rADCTSC=rADCTSC|(1<<8);//Detectstylusupinterruptsignal.
10、如果發(fā)生中斷,不做任何操作,只打印出一句觸摸筆抬起中斷信息
while(1)//tocheckPen-upstate
{
if(rSUBSRCPND&(BIT_SUB_TC))//checkifADCisfinishedwithinterruptbit
{
Uart_Printf("StylusUpInterrupt~!");
break;//ifStylusisup(1)state
}
}
Uart_Printf("count=dXP=d,YP=d",count++,xdata,ydata);
11、觸摸筆抬起之后,把得到的x,y坐標值發(fā)送給PC機,顯示出具體數(shù)值
三)再次設置觸摸屏為等待中斷模式,等待下次觸摸屏被按下
rADCTSC=0xd3;//Waitingforinterrupt
ClearSubPending(BIT_SUB_TC);
ClearPending(BIT_ADC);
EnableSubIrq(BIT_SUB_TC);
EnableIrq(BIT_ADC);
示例程序如下:
#defineGLOBAL_CLK1
#include
#include
#include"def.h"
#include"option.h"
#include"2440addr.h"
#include"2440lib.h"
#include"2440slib.h"
#include"mmu.h"
#include"profile.h"
#include"memtest.h"
#defineADC_FREQ2500000
intcount=0;
volatileU32preScaler;
intxdata,ydata;
voidTest_Touchpanel(void);
staticvoid__irqAdcTsAuto(void);
staticvoidcal_cpu_bus_clk(void);
voidSet_Clk(void);
voiddelay(inttimes)
{
inti,j;
for(i=0;i
for(j=0;j<400;j++);
}
intMain(void)
{
intScom=0;
Set_Clk();
Uart_Init(0,115200);
Uart_Select(Scom);
Test_Touchpanel();
while(1);
return0;
}
評論