真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

c語(yǔ)言鋸齒波發(fā)生函數(shù) 函數(shù)發(fā)生器鋸齒波

基于單片機(jī)的波形發(fā)生器設(shè)計(jì),C語(yǔ)言程序,有個(gè)很小的問(wèn)題,求解決

void juchi() //鋸齒波函數(shù)

我們提供的服務(wù)有:成都網(wǎng)站設(shè)計(jì)、成都網(wǎng)站建設(shè)、微信公眾號(hào)開發(fā)、網(wǎng)站優(yōu)化、網(wǎng)站認(rèn)證、互助ssl等。為上千多家企事業(yè)單位解決了網(wǎng)站和推廣的問(wèn)題。提供周到的售前咨詢和貼心的售后服務(wù),是有科學(xué)管理、有技術(shù)的互助網(wǎng)站制作公司

{ uchar i;

for(i=0;i255;i++)

{DAC0832=i;

dump(10+p);} //延時(shí)實(shí)現(xiàn)頻率可調(diào),p是變量

i=0x00; }

void zhengxian()

{

for(i=0;i256;i++)//i沒(méi)有聲明 上面那個(gè)程序都有uchar i; uchar類型的最大值也只有255

{DAC0832=sin[i];//查表取值

dump(2+p);

}

i=0;

}

//最簡(jiǎn)單的方法就是復(fù)制juchi這個(gè)函數(shù)

//把DAC0832=i;改成DAC832=sin[i];就能看到效果了

void zhengxian() //正弦波函數(shù)

{ uchar i;

for(i=0;i255;i++)

{DAC0832=sin[i];

dump(10+p);} //延時(shí)實(shí)現(xiàn)頻率可調(diào),p是變量

i=0x00; }

DAC0832的輸出形式

如圖9-58所示, 由運(yùn)算放大器進(jìn)行電流→電壓轉(zhuǎn)換,使用內(nèi)部反饋電阻。輸出電壓值VOUT和輸入數(shù)字量D的關(guān)系:

VOUT = - VREF ×D/256

D = 0~255, VOUT = 0 ~ - VREF ×255/256

VREF = -5V, VOUT =0~5×(255/256)V

VREF = +5V, VOUT = 0 ~ -5×(255/256)V 如果實(shí)際應(yīng)用系統(tǒng)中要求輸出模擬電壓為雙極性,則需要用轉(zhuǎn)換電路實(shí)現(xiàn)。如圖9-59所示。

其中 R2=R3=2R1

VOUT= 2×VREF×D/256 -VREF= (2D/256-1)VREF

D = 0, VOUT= -VREF;

D = 128, VOUT= 0;

D = 255, VOUT= (2×255/256-1)×VREF= (254/255)VREF

即:輸入數(shù)字為0~255時(shí),輸出電壓在- VREF ~+ VREF之間變化。

1. 運(yùn)算放大器

運(yùn)算放大器有三個(gè)特點(diǎn):

⑴開環(huán)放大倍數(shù)非常高,一般為幾千,甚至可高達(dá)10萬(wàn)。在正常情況下,運(yùn)算放大器所需要的輸入電壓非常小。

⑵輸入阻抗非常大。運(yùn)算放大器工作時(shí),輸入端相當(dāng)于一個(gè)很小的電壓加在一個(gè)很大的輸入阻抗上,所需要的輸入電流也極小。

⑶輸出阻抗很小,所以,它的驅(qū)動(dòng)能力非常大。

2.由電阻網(wǎng)絡(luò)和運(yùn)算放大器構(gòu)成的D/A轉(zhuǎn)換器

利用運(yùn)算放大器各輸入電流相加的原理,可以構(gòu)成如圖10.7所示的、由電阻網(wǎng)絡(luò)和運(yùn)算放大器組成的、最簡(jiǎn)單的4位D/A轉(zhuǎn)換器。圖中,V0是一個(gè)有足夠精度的標(biāo)準(zhǔn)電源。運(yùn)算放大器輸入端的各支路對(duì)應(yīng)待轉(zhuǎn)換資料的D0,D1,…,Dn-1位。各輸入支路中的開關(guān)由對(duì)應(yīng)的數(shù)字元值控制,如果數(shù)字元為1,則對(duì)應(yīng)的開關(guān)閉合;如果數(shù)字為0,則對(duì)應(yīng)的開關(guān)斷開。各輸入支路中的電阻分別為R,2R,4R,…這些電阻稱為權(quán)電阻。

假設(shè),輸入端有4條支路。4條支路的開關(guān)從全部斷開到全部閉合,運(yùn)算放大器可以得到16種不同的電流輸入。這就是說(shuō),通過(guò)電阻網(wǎng)絡(luò),可以把0000B~1111B轉(zhuǎn)換成大小不等的電流,從而可以在運(yùn)算放大器的輸出端得到相應(yīng)大小不同的電壓。如果數(shù)字0000B每次增1,一直變化到1111B,那么,在輸出端就可得到一個(gè)0~V0電壓幅度的階梯波形。

3.采用T型電阻網(wǎng)絡(luò)的D/A轉(zhuǎn)換器

從圖10.7可以看出,在D/A轉(zhuǎn)換中采用獨(dú)立的權(quán)電阻網(wǎng)絡(luò),對(duì)于一個(gè)8位二進(jìn)制數(shù)的D/A轉(zhuǎn)換器,就需要R,2R,4R,…,128R共8個(gè)不等的電阻,最大電阻阻值是最小電阻阻值的128倍,而且對(duì)這些電阻的精度要求比較高。如果這樣的話,從工藝上實(shí)現(xiàn)起來(lái)是很困難的。所以,n個(gè)如此獨(dú)立輸入支路的方案是不實(shí)用的。

在DAC電路結(jié)構(gòu)中,最簡(jiǎn)單而實(shí)用的是采用T型電阻網(wǎng)絡(luò)來(lái)代替單一的權(quán)電阻網(wǎng)絡(luò),整個(gè)電阻網(wǎng)絡(luò)只需要R和2R兩種電阻。在集成電路中,由于所有的組件都做在同一芯片上,電阻的特性可以做得很相近,而且精度與誤差問(wèn)題也可以得到解決。

圖10.8是采用T型電阻網(wǎng)絡(luò)的4位D/A轉(zhuǎn)換器。4位元待轉(zhuǎn)換資料分別控制4條支路中開關(guān)的倒向。在每一條支路中,如果(資料為0)開頭倒向左邊,支路中的電阻就接到地;如果(資料為1)開關(guān)倒向右邊,電阻就接到虛地。所以,不管開關(guān)倒向哪一邊,都可以認(rèn)為是接“地”。不過(guò),只有開關(guān)倒向右邊時(shí),才能給運(yùn)算放大器輸入端提供電流。

T型電阻網(wǎng)絡(luò)中,節(jié)點(diǎn)A的左邊為兩個(gè)2R的電阻并聯(lián),它們的等效電阻為R,節(jié)點(diǎn)B的左邊也是兩個(gè)2R的電阻并聯(lián),它們的等效電阻也是R,…,依次類推,最后在D點(diǎn)等效于一個(gè)數(shù)值為R的電阻接在參考電壓VREF上。這樣,就很容易算出,C點(diǎn)、B點(diǎn)、A點(diǎn)的電位分別為-VREF/2,-VREF/4,-VREF/8。

在清楚了電阻網(wǎng)絡(luò)的特點(diǎn)和各節(jié)點(diǎn)的電壓之后,再來(lái)分析一下各支路的電流值。開關(guān)S3,S2,S1,S0分別代表對(duì)應(yīng)的1位二進(jìn)制數(shù)。任一資料位Di=1,表示開關(guān)Si倒向右邊;Di=0,表示開關(guān)Si倒向左邊,接虛地,無(wú)電流。當(dāng)右邊第一條支路的開關(guān)S3倒向右邊時(shí),運(yùn)算放大器得到的輸入電流為-VREF/(2R),同理,開關(guān)S2,S1,S0倒向右邊時(shí),輸入電流分別為-VREF/(4R),-VREF/(8R),-VREF/(16R)。

如果一個(gè)二進(jìn)制數(shù)據(jù)為1111,運(yùn)算放大器的輸入電流

I=-VREF/(2R)-VREF/(4R)-VREF/(8R)-VREF/(16R)

=-VREF/(2R)(20+2-1+2-2+2-3)

=-VREF/(24R)(23+22+21+20)

相應(yīng)的輸出電壓

V0=IR0=-VREFR0(24R)(23+22+21+20)

將資料推廣到n位,輸出模擬量與輸入數(shù)字量之間關(guān)系的一般表達(dá)式為:

V0=-VREFR0/(2nR)(Dn-12n-1+Dn-2 2n-2+…+D121+D020) (Di=1或0)

上式表明,輸出電壓V0除了和待轉(zhuǎn)換的二進(jìn)制數(shù)成比例外,還和網(wǎng)絡(luò)電阻R、運(yùn)算放大器反饋電阻R0、標(biāo)準(zhǔn)參考電壓VREF有關(guān)。

2. D/A轉(zhuǎn)換器性能參數(shù)

在實(shí)現(xiàn)D/A轉(zhuǎn)換時(shí),主要涉及下面幾個(gè)性能參數(shù)。

⑴分辨率。分辨率是指最小輸出電壓(對(duì)應(yīng)于輸入數(shù)字量最低位增1所引起的輸出電壓增量)和最大輸出電壓(對(duì)應(yīng)于輸入數(shù)字量所有有效位全為1時(shí)的輸出電壓)之比,

例如,4位DAC的分辨率為1/(16-1)=1/15=6.67%(分辨率也常用百分比來(lái)表示)。8位DAC的分辨率為1/255=0.39%。顯然,位數(shù)越多,分辨率越高。

⑵轉(zhuǎn)換精度。如果不考慮D/A轉(zhuǎn)換的誤差,DAC轉(zhuǎn)換精度就是分辨率的大小,因此,要獲得高精度的D/A轉(zhuǎn)換結(jié)果,首先要選擇有足夠高分辨率的DAC。

D/A轉(zhuǎn)換精度分為絕對(duì)和相對(duì)轉(zhuǎn)換精度,一般是用誤差大小表示。DAC的轉(zhuǎn)換誤差包括零點(diǎn)誤差、漂移誤差、增益誤差、噪聲和線性誤差、微分線性誤差等綜合誤差。

絕對(duì)轉(zhuǎn)換精度是指滿刻度數(shù)字量輸入時(shí),模擬量輸出接近理論值的程度。它和標(biāo)準(zhǔn)電源的精度、權(quán)電阻的精度有關(guān)。相對(duì)轉(zhuǎn)換精度指在滿刻度已經(jīng)校準(zhǔn)的前提下,整個(gè)刻度范圍內(nèi),對(duì)應(yīng)任一模擬量的輸出與它的理論值之差。它反映了DAC的線性度。通常,相對(duì)轉(zhuǎn)換精度比絕對(duì)轉(zhuǎn)換精度更有實(shí)用性。

相對(duì)轉(zhuǎn)換精度一般用絕對(duì)轉(zhuǎn)換精度相對(duì)于滿量程輸出的百分?jǐn)?shù)來(lái)表示,有時(shí)也用最低位(LSB)的幾分之幾表示。例如,設(shè)VFS為滿量程輸出電壓5V,n位DAC的相對(duì)轉(zhuǎn)換精度為±0.1%,則最大誤差為±0.1%VFS=±5mV;若相對(duì)轉(zhuǎn)換精度為±1/2LSB,LSB=1/2n,則最大相對(duì)誤差為±1/2n+1VFS。

⑶非線性誤差。D/A轉(zhuǎn)換器的非線性誤差定義為實(shí)際轉(zhuǎn)換特性曲線與理想特性曲線之間的最大偏差,并以該偏差相對(duì)于滿量程的百分?jǐn)?shù)度量。轉(zhuǎn)換器電路設(shè)計(jì)一般要求非線性誤差不大于±1/2LSB。

⑷轉(zhuǎn)換速率/建立時(shí)間。轉(zhuǎn)換速率實(shí)際是由建立時(shí)間來(lái)反映的。建立時(shí)間是指數(shù)字量為滿刻度值(各位全為1)時(shí),DAC的模擬輸出電壓達(dá)到某個(gè)規(guī)定值(比如,90%滿量程或±1/2LSB滿量程)時(shí)所需要的時(shí)間。

建立時(shí)間是D/A轉(zhuǎn)換速率快慢的一個(gè)重要參數(shù)。很顯然,建立時(shí)間越大,轉(zhuǎn)換速率越低。不同型號(hào)DAC的建立時(shí)間一般從幾個(gè)毫微秒到幾個(gè)微秒不等。若輸出形式是電流,DAC的建立時(shí)間是很短的;若輸出形式是電壓,DAC的建立時(shí)間主要是輸出運(yùn)算放大器所需要的響應(yīng)時(shí)間。

10.3.3 DAC0832及接口電路

DAC0832是美國(guó)資料公司研制的8位雙緩沖器D/A轉(zhuǎn)換器。芯片內(nèi)帶有資料鎖存器,可與數(shù)據(jù)總線直接相連。電路有極好的溫度跟隨性,使用了COMS電流開關(guān)和控制邏輯而獲得低功耗、低輸出的泄漏電流誤差。芯片采用R-2RT型電阻網(wǎng)絡(luò),對(duì)參考電流進(jìn)行分流完成D/A轉(zhuǎn)換。轉(zhuǎn)換結(jié)果以一組差動(dòng)電流IOUT1和IOUT2輸出。

1.DAC0832的內(nèi)部結(jié)構(gòu)

DAC0832中有兩級(jí)鎖存器,第一級(jí)鎖存器稱為輸入寄存器,它的鎖存信號(hào)為ILE;第二級(jí)鎖存器稱為DAC寄存器,它的鎖存信號(hào)為傳輸控制信號(hào) 。因?yàn)橛袃杉?jí)鎖存器,DAC0832可以工作在雙緩沖器方式,即在輸出模擬信號(hào)的同時(shí)采集下一個(gè)數(shù)字量,這樣能有效地提高轉(zhuǎn)換速度。此外,兩級(jí)鎖存器還可以在多個(gè)D/A轉(zhuǎn)換器同時(shí)工作時(shí),利用第二級(jí)鎖存信號(hào)來(lái)實(shí)現(xiàn)多個(gè)轉(zhuǎn)換器同步輸出。

ILE為高電平、WR1 和 CS為低電平時(shí), LE1為高電平,輸入寄存器的輸出跟隨輸入而變化;此后,當(dāng)WR1 由低變高時(shí), LE1為低電平,資料被鎖存到輸入寄存器中,這時(shí)的輸入寄存器的輸出端不再跟隨輸入資料的變化而變化。對(duì)第二級(jí)鎖存器來(lái)說(shuō), WR2和XFER 同時(shí)為低電平時(shí), LE2為高電平,DAC寄存器的輸出跟隨其輸入而變化;此后,當(dāng)WR2 由低變高時(shí), LE2變?yōu)榈碗娖剑瑢⑤斎爰拇嫫鞯馁Y料鎖存到DAC寄存器中。

2. DAC0832的引腳特性

DAC0832是20引腳的雙列直插式芯片。各引腳的特性如下:

CS——片選信號(hào),和允許鎖存信號(hào)ILE組合來(lái)決定 是否起作用,低有效。

ILE——允許鎖存信號(hào),高有效。

WR1——寫信號(hào)1,作為第一級(jí)鎖存信號(hào),將輸入資料鎖存到輸入寄存器(此時(shí), 必須和 、ILE同時(shí)有效),低有效。

WR2——寫信號(hào)2,將鎖存在輸入寄存器中的資料送到DAC寄存器中進(jìn)行鎖存(此時(shí),傳輸控制信號(hào) 必須有效)低有效。

XFER——傳輸控制信號(hào),低有效。

DI7~DI0——8位數(shù)據(jù)輸入端。

IOUT1——模擬電流輸出端1。當(dāng)DAC寄存器中全為1時(shí),輸出電流最大,當(dāng)DAC寄存器中全為0時(shí),輸出電流為0。

IOUT2——模擬電流輸出端2。IOUT1+IOUT2=常數(shù)。

Rfb——反饋電阻引出端。DAC0832內(nèi)部已經(jīng)有反饋電阻,所以,RFB端可以直接接到外部運(yùn)算放大器的輸出端。相當(dāng)于將反饋電阻接在運(yùn)算放大器的輸入端和輸出端之間。

VREF——參考電壓輸入端??山与妷悍秶鸀椤?0V。外部標(biāo)準(zhǔn)電壓通過(guò)VREF與T型電阻網(wǎng)絡(luò)相連。

VCC——芯片供電電壓端。范圍為+5V~+15V,最佳工作狀態(tài)是+15V。

AGND——模擬地,即模擬電路接地端。

DGND——數(shù)字地,即數(shù)字電路接地端。

3.DAC0832的工作方式

DAC0832進(jìn)行D/A轉(zhuǎn)換,可以采用兩種方法對(duì)數(shù)據(jù)進(jìn)行鎖存。

第一種方法是使輸入寄存器工作在鎖存狀態(tài),而DAC寄存器工作在直通狀態(tài)。具體地說(shuō),就是使 和 都為低電平,DAC寄存器的鎖存選通端得不到有效電平而直通;此外,使輸入寄存器的控制信號(hào)ILE處于高電平、 處于低電平,這樣,當(dāng) 端來(lái)一個(gè)負(fù)脈沖時(shí),就可以完成1次轉(zhuǎn)換。

第二種方法是使輸入寄存器工作在直通狀態(tài),而DAC寄存器工作在鎖存狀態(tài)。就是使 和 為低電平,ILE為高電平,這樣,輸入寄存器的鎖存選通信號(hào)處于無(wú)效狀態(tài)而直通;當(dāng)WR2 和XFER 端輸入1個(gè)負(fù)脈沖時(shí),使得DAC寄存器工作在鎖存狀態(tài),提供鎖存數(shù)據(jù)進(jìn)行轉(zhuǎn)換。

根據(jù)上述對(duì)DAC0832的輸入寄存器和DAC寄存器不同的控制方法,DAC0832有如下3種工作方式:

⑴單緩沖方式。單緩沖方式是控制輸入寄存器和DAC寄存器同時(shí)接收資料,或者只用輸入寄存器而把DAC寄存器接成直通方式。此方式適用只有一路模擬量輸出或幾路模擬量異步輸出的情形。

⑵雙緩沖方式。雙緩沖方式是先使輸入寄存器接收資料,再控制輸入寄存器的輸出資料到DAC寄存器,即分兩次鎖存輸入資料。此方式適用于多個(gè)D/A轉(zhuǎn)換同步輸出的情節(jié)。

⑶直通方式。直通方式是資料不經(jīng)兩級(jí)鎖存器鎖存,即 CS*,XFER* ,WR1* ,WR2* 均接地,ILE接高電平。此方式適用于連續(xù)反饋控制線路和不帶微機(jī)的控制系統(tǒng),不過(guò)在使用時(shí),必須通過(guò)另加I/O接口與CPU連接,以匹配CPU與D/A轉(zhuǎn)換。

4. DAC0832的應(yīng)用舉例

⑴DAC0832實(shí)現(xiàn)一次D/A轉(zhuǎn)換,可以采用下面程序段。設(shè)定要轉(zhuǎn)換的數(shù)據(jù)放在1000H單元中。

MOV BX,100H

MOV AL,[BX] ;取轉(zhuǎn)換資料

OUT DX,AL

⑵在實(shí)際應(yīng)用中,經(jīng)常需要用到一個(gè)線性增長(zhǎng)的電壓去控制某一個(gè)檢測(cè)過(guò)程,或者作為掃描電壓去控制一個(gè)電子束的移動(dòng)。執(zhí)行下面的程序段,利用D/A轉(zhuǎn)換器產(chǎn)生一個(gè)鋸齒波電壓,實(shí)現(xiàn)此類控制作用。

MOV DX,PORTA ;PORTA為D/A轉(zhuǎn)換器端口地址

MOV AL,OFFH ;置初值

ROTAT:INC AL

OUT DX,AL ;往D/A轉(zhuǎn)換器輸出資料

CALL DELP ;調(diào)用延遲子程序

JMP ROTAT

DELY: MOV CX, DATA ;置延遲常數(shù)DATA

DELY1: LOOP DELY1

RET

如果需要一個(gè)負(fù)向的鋸齒波,只要將指令I(lǐng)NC AL改成DEC AL就可以了。

⑶從兩個(gè)不相關(guān)的文件中輸出一批X-Y資料,驅(qū)動(dòng)X-Y記錄儀,或者控制加工復(fù)雜零件的走刀(X軸)和進(jìn)刀(Y軸)。這些在控制過(guò)程中是很有用的。下面程序驅(qū)動(dòng)X-Y記錄儀的100點(diǎn)輸出,并用軟件驅(qū)動(dòng)記錄儀的抬筆和放筆控制。

MOV SI, XDATA ;X軸資料指針→SI

MOV DI, YDATA ;Y軸資料指針→DI

MOV CX, 100

WE0: MOV AL,[SI]

OUT PORTX, AL ;往X軸的D/A轉(zhuǎn)換器輸出資料

MOV AL,[DI]

OUT PORTY,AL;往Y軸的D/A轉(zhuǎn)換器輸出資料

CALL DELY1 ;調(diào)延遲子程序1,等待筆移動(dòng)

MOV AL,01H

OUT PORTM,AL;輸出升脈沖,控制筆放下

CALL DELY2 ;調(diào)延遲子程序2,等待完成

MOV AL,00H

OUT PORTM,AL;輸出降脈沖,控制筆抬起

CALL DELY2 ;調(diào)延遲子程序2,等待完成

INC SI

INC DI

LOOP WE0

HLT

DELY1:┇

RET

DELY2:┇

RET

XDATA DB…

YDATA DB…

(4).利用C語(yǔ)言編程:

#pragma db oe sb

#i ncludereg51.h

#i ncludeabsacc.h

#define DAC0832 XBYTE[0x7fff] /* 定義DAC0832端口地址 */

#define uchar unsigned char

void delay(uchar t) { /* 延時(shí)函數(shù) */

while(t--);

}

void saw(void) { /* 鋸齒波發(fā)生函數(shù) */

uchar i;

for (i=0;i255;i++) {

DAC0832=i;

}

}

void square(void) { /* 方波發(fā)生函數(shù) */

DAC0832=0x00;

delay(0x10);

DAC0832=0xff;

delay(0x10);

}

void main(void) {

uchar i,j;

i=j=0xff;

while(i--) {

saw(); /* 產(chǎn)生一段鋸齒波 */

}

while(j--) {

square(); /* 產(chǎn)生一段方波 */

}

}

at89c52產(chǎn)生三角波,鋸齒波,方波,要求用c語(yǔ)言寫,芯片是D0832

#includereg52.h

#define uchar unsigned char

#define uint unsigned int

//#define Fosc 24000000/12000000 //12分頻后的頻率

#define DAdata P0//DA數(shù)據(jù)端口

sbit DA_S1= P2^0; // 控制DAC0832的8位輸入寄存器,僅當(dāng)都為0時(shí),可以輸出數(shù)據(jù)(處于直通狀態(tài)),否則,輸出將被鎖存

sbit DA_S2= P2^1; // 控制DAC0832的8位DAC寄存器,僅當(dāng)都為0時(shí),可以輸出數(shù)據(jù)(處于直通狀態(tài)),否則,輸出將被鎖存

sbit key= P3^2;

uchar wavecount; //'抽點(diǎn)'計(jì)數(shù)

uchar THtemp,TLtemp;//傳遞頻率的中間變量

//uint T_temp;

uchar judge=1; //在方波輸出函數(shù)中用于簡(jiǎn)單判別作用

uchar waveform; //當(dāng)其為0、1、2時(shí),分別代表三種波

uchar code freq_unit[4]={10,50,200,10}; //三種波的頻率單位 sawtooth

uchar idata wavefreq[4]={1,1,1,1}; //給每種波定義一個(gè)數(shù)組單元,用于存放單位頻率的個(gè)數(shù)

uchar code lcd_hang1[]={"Sine Wave " "Triangle Wave " "Square Wave " "sawtooth Wave ""Select Wave: " "press No.1 key! "};

uchar idata lcd_hang2[16]={"f= Hz "};

/*uchar code wave_freq_adjust[]={ //頻率調(diào)整中間值

0xff,0xb8,0x76,0x56,0x43,0x37,0x2e,0x26,0x20,0x1c, //正弦波頻率調(diào)整中間值

0xff,0x8e,0x5a,0x41,0x32,0x28,0x20,0x1b,0x17,0x0e,//三角波頻率調(diào)整中間值

0xff,0x8e,0x5a,0x41,0x32,0x28,0x20,0x1b,0x17,0x0e};

uint code wave_freq_adjust[]={ //頻率調(diào)整中間值

380,184,118,86,67,55,46,28,38,32,

295,142, 90,65,50,40,32,27,23,14,

295,142, 90,65,50,40,32,27,23,14}; */

/*uchar code waveTH[]={

0xfc,0xfe,0xfe,0xff,0xff,0xff,0xff,0xff,0xff,0xff,

0xfc,0xfe,0xfe,0xff,0xff,0xff,0xff,0xff,0xff,0xff,

0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff};

uchar code waveTL[]={

0xf2,0x78,0xfb,0x3c,0x63,0x7d,0x8f,0x9d,0xa8,0xb1,

0x17,0x0b,0xb2,0x05,0x37,0x58,0x70,0x82,0x90,0x9b,

0x4d,0xa7,0xc4,0xd3,0xdc,0xe2,0xe6,0xea,0xec,0xee};*/

/***********這兩組數(shù)組很重要,需要根據(jù)波形來(lái)調(diào)試,選擇合適的值,使輸出波形達(dá)到頻率要求************/

uchar code waveTH[]={

0xfd,0xfe,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,

0xfd,0xfe,0xff,0xff,0xff,0xff,0xff,0xff,0xff,0xff,

0xec,0xf6,0xf9,0xfb,0xfc,0xfc,0xfd,0xfd,0xfd,0xfe};

uchar code waveTL[]={

0x06,0x8a,0x10,0x4e,0x78,0x93,0xa8,0xb3,0xbe,0xc6, //正弦波頻率調(diào)整中間值

0xac,0xde,0x48,0x7a,0x99,0xaf,0xbb,0xc8,0xd0,0xde,//三角波頻率調(diào)整中間值

0x88,0x50,0x90,0x32,0x34,0xbe,0x4a,0xa3,0xe5,0x2c};

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

uchar code triangle_tab[]={ //每隔數(shù)字8,采取一次

0x00,0x08,0x10,0x18,0x20,0x28,0x30,0x38,0x40,0x48,0x50,0x58,0x60,0x68,0x70,0x78,

0x80,0x88,0x90,0x98,0xa0,0xa8,0xb0,0xb8,0xc0,0xc8,0xd0,0xd8,0xe0,0xe8,0xf0,0xf8,0xff,

0xf8,0xf0,0xe8,0xe0,0xd8,0xd0,0xc8,0xc0,0xb8,0xb0,0xa8,0xa0,0x98,0x90,0x88,0x80,

0x78,0x70,0x68,0x60,0x58,0x50,0x48,0x40,0x38,0x30,0x28,0x20,0x18,0x10,0x08,0x00};

uchar code sine_tab[256]={

//輸出電壓從0到最大值(正弦波1/4部分)

0x80,0x83,0x86,0x89,0x8d,0x90,0x93,0x96,0x99,0x9c,0x9f,0xa2,0xa5,0xa8,0xab,0xae,0xb1,0xb4,0xb7,0xba,0xbc,

0xbf,0xc2,0xc5,0xc7,0xca,0xcc,0xcf,0xd1,0xd4,0xd6,0xd8,0xda,0xdd,0xdf,0xe1,0xe3,0xe5,0xe7,0xe9,0xea,0xec,

0xee,0xef,0xf1,0xf2,0xf4,0xf5,0xf6,0xf7,0xf8,0xf9,0xfa,0xfb,0xfc,0xfd,0xfd,0xfe,0xff,0xff,0xff,0xff,0xff,0xff,

//輸出電壓從最大值到0(正弦波1/4部分)

0xff,0xff,0xff,0xff,0xff,0xff,0xfe,0xfd,0xfd,0xfc,0xfb,0xfa,0xf9,0xf8,0xf7,0xf6,0xf5,0xf4,0xf2,0xf1,0xef,

0xee,0xec,0xea,0xe9,0xe7,0xe5,0xe3,0xe1,0xde,0xdd,0xda,0xd8,0xd6,0xd4,0xd1,0xcf,0xcc,0xca,0xc7,0xc5,0xc2,

0xbf,0xbc,0xba,0xb7,0xb4,0xb1,0xae,0xab,0xa8,0xa5,0xa2,0x9f,0x9c,0x99 ,0x96,0x93,0x90,0x8d,0x89,0x86,0x83,0x80,

//輸出電壓從0到最小值(正弦波1/4部分)

0x80,0x7c,0x79,0x76,0x72,0x6f,0x6c,0x69,0x66,0x63,0x60,0x5d,0x5a,0x57,0x55,0x51,0x4e,0x4c,0x48,0x45,0x43,

0x40,0x3d,0x3a,0x38,0x35,0x33,0x30,0x2e,0x2b,0x29,0x27,0x25,0x22,0x20,0x1e,0x1c,0x1a,0x18,0x16 ,0x15,0x13,

0x11,0x10,0x0e,0x0d,0x0b,0x0a,0x09,0x08,0x07,0x06,0x05,0x04,0x03,0x02,0x02,0x01,0x00,0x00,0x00,0x00,0x00,0x00,

//輸出電壓從最小值到0(正弦波1/4部分)

0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x02 ,0x02,0x03,0x04,0x05,0x06,0x07,0x08,0x09,0x0a,0x0b,0x0d,0x0e,0x10,

0x11,0x13,0x15 ,0x16,0x18,0x1a,0x1c,0x1e,0x20,0x22,0x25,0x27,0x29,0x2b,0x2e,0x30,0x33,0x35,0x38,0x3a,0x3d,

0x40,0x43,0x45,0x48,0x4c,0x4e,0x51,0x55,0x57,0x5a,0x5d,0x60,0x63,0x66 ,0x69,0x6c,0x6f,0x72,0x76,0x79,0x7c,0x80};

uchar code sawtooth_tab[]={

0x00,0x02,0x04,0x06,0x08,0x0a,0x0c,0x0e,0x10,0x12,0x14,0x16,0x18,0x1a,0x1c,0x1e,0x20,0x22,0x24,0x26,0x28,0x2a,0x2c,0x2e,

0x30,0x32,0x34,0x36,0x38,0x3a,0x3c,0x3e,0x40,0x42,0x44,0x46,0x48,0x4a,0x4c,0x4e,0x50,0x52,0x54,0x56,0x58,0x5a,0x5c,0x5e,

0x60,0x62,0x64,0x66,0x68,0x6a,0x6c,0x6e,0x70,0x72,0x74,0x76,0x78,0x7a,0x7c,0x7e,0x80,0x82,0x84,0x86,0x88,0x8a,0x8c,0x8e};

//0x90,0x92,0x94,0x96,0x98,0x9a,0x9c,0x9e,0xa0,0xa2,0xa4,0xa6,0xa8,0xaa,0xac,0xae,0xb0,0xb2,0xb4,0xb6,0xb8,0xba,0xbc,0xbe,

//0xc0,0xc2,0xc4,0xc6,0xc8,0xca,0xcc,0xce,0xd0,0xd2,0xd4,0xd6,0xd8,0xda,0xdc,0xde,0xe0,0xe2,0xe4,0xe6,0xe8,0xea,0xec,0xee};

void delay(uchar z)

{

uint x,y;

for(x=z;x0;x--)

for(y=110;y0;y--);

}

void triangle_out()//三角波輸出

{

DAdata=triangle_tab[wavecount++];

if(wavecount64) wavecount=0;

DA_S1=0; //打開8位輸入寄存器

DA_S1=1; //關(guān)閉8位輸入寄存器

}

void sine_out() //正弦波輸出

{

DAdata=sine_tab[wavecount++];

DA_S1=0; //打開8位輸入寄存器

DA_S1=1; //關(guān)閉8位輸入寄存器

}

void square_out() //方波輸出

{

judge=~judge;

if(judge==1) DAdata=0xff;

else DAdata=0x00;

DA_S1=0; //打開8位輸入寄存器

DA_S1=1; //關(guān)閉8位輸入寄存器

}

void sawtooth_out() //鋸齒波輸出

{

DAdata=sawtooth_tab[wavecount++];

if(wavecount71) wavecount=0;

DA_S1=0; //打開8位輸入寄存器

DA_S1=1; //關(guān)閉8位輸入寄存器

}

/************1602液晶的相關(guān)函數(shù)*************/

#define lcd_ports P1

sbit rs=P2^2;

sbit rw=P2^3;

sbit lcden=P2^4;

void write_com(uchar com)

{

rs=0;//置零,表示寫指令

lcden=0;

lcd_ports=com;

delay(5);

lcden=1;

delay(5);

lcden=0;

}

void write_date(uchar date)

{

rs=1;//置1,表示寫數(shù)據(jù)(在指令所指的地方寫數(shù)據(jù))

lcden=0;

lcd_ports=date;

delay(5);

lcden=1;

delay(5);

lcden=0;

}

void disp_lcd(uchar addr,uchar *temp1)

{

uchar num;

write_com(addr);

delay(1); //延時(shí)一會(huì)兒???

for(num=0;num16;num++)

{

write_date(temp1[num]);//或者這樣寫write_date(*(temp1+num));

delay(1);

}

}

void init_lcd()

{

//uchar num;

lcden=0; //可有可無(wú)???

rw=0; //初始化一定要設(shè)置為零,表示寫數(shù)據(jù)

write_com(0x38); //使液晶顯示點(diǎn)陣,為下面做準(zhǔn)備

write_com(0x0c); //初始設(shè)置

write_com(0x06); //初始設(shè)置

write_com(0x01); //清零

write_com(0x80); //使指針指向第一行第一格

disp_lcd(0x80,lcd_hang1[4*16]); //在第一行顯示

disp_lcd(0xc0,lcd_hang1[5*16]); //在第二行顯示

/*for(num=0;num16;num++)

{

write_date(table[num]);

delay(5);

}

write_com(0x80+0x40); //給指針重新賦值,使之指向第二行第一格

for(num=0;num16;num++)

{

write_date(table1[num]);

delay(5);

} */

/*TMOD=0x01; //選用定時(shí)方式1

TH0=(65536-50000)/256; //賦初值

TL0=(65536-50000)%256;//

EA=1;//開總中斷

ET0=1; //開定時(shí)器中斷

TR0=1; //啟動(dòng)定時(shí)器*/

}

/********************1602液晶函數(shù)聲明結(jié)束*********************/

void main()

{

uchar i=0;

DA_S2=0; //使DAC寄存器處于直通狀態(tài)

DAdata=0;

DA_S1=1; //關(guān)閉8位輸入寄存器

init_lcd();

waveform=0;

TMOD=0x01; //設(shè)置定時(shí)器0為16位工作方式

IT0=1; //設(shè)置外部中斷0為下降沿觸發(fā)

ET0=1; //開定時(shí)器中斷

EX0=1;

EA=1;

while(1)

{

//DAout(0xff); //可輸出TTL波形

//DAout(0x80);

//T_temp=32;

}

}

void timer0() interrupt 1

{

TH0=THtemp;

TL0=TLtemp;

if(waveform==0) sine_out();

else if(waveform==1) triangle_out();

else if(waveform==2) square_out();

else if(waveform==3) sawtooth_out(); //tian jian

}

void key_int0() interrupt 0

{

uchar keytemp;

uint total_freq; //總頻率

EA=0; TR0=0; //關(guān)總中斷與定時(shí)器

delay(5); //延時(shí)夠嗎???

if(key==0) //確實(shí)有按鍵按下而引發(fā)中斷

{

keytemp=P30xf0; //獲取P3口高四位的值

switch(keytemp)

{

case 0xe0://選擇波形

waveform++;

if(waveform3) waveform=0; //jiang 2 gai wei 3

break;

case 0xd0: //頻率按規(guī)定單位依次增加

wavefreq[waveform]++;

if(wavefreq[waveform]10) wavefreq[waveform]=1; // /*這邊要用“10”,因?yàn)樗取?11”可靠

break; // 性更高,使加數(shù)有個(gè)上限,不會(huì)一直加下去*/

case 0xb0: //頻率按規(guī)定單位依次衰減

wavefreq[waveform]--;

if(wavefreq[waveform]1) wavefreq[waveform]=10; //這邊要用“1”,因?yàn)樗取?0”可靠性更高

break;

case 0x70: //TTL輸出

DA_S2=1; //使DAC寄存器關(guān)閉

break;

}

THtemp=waveTH[waveform*10+(wavefreq[waveform]-1)]; //方括號(hào)中選取第幾個(gè)數(shù)后,并把該值賦給T_temp

TLtemp=waveTL[waveform*10+(wavefreq[waveform]-1)];

total_freq= wavefreq[waveform] * freq_unit[waveform]; //求輸出頻率(個(gè)數(shù)*單位)

lcd_hang2[5]=total_freq%10+0x30; //在液晶中顯示個(gè)位,(0x30 在液晶顯示中表示數(shù)字0)

total_freq/=10; lcd_hang2[4]=total_freq%10+0x30; //在液晶中顯示時(shí)十位

total_freq/=10; lcd_hang2[3]=total_freq%10+0x30; //在液晶中顯示時(shí)百位

total_freq/=10; lcd_hang2[2]=total_freq%10+0x30; //在液晶中顯示時(shí)千位

disp_lcd(0x80,lcd_hang1[waveform*16]); //在第一行顯示

disp_lcd(0xc0,lcd_hang2); //在第二行顯示

}

wavecount=0; //'抽點(diǎn)'計(jì)數(shù)清零

while(!key);

EA=1; TR0=1; //開啟總中斷與定時(shí)器

產(chǎn)生三角波數(shù)據(jù)值的C程序,

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

/* 信號(hào)發(fā)生器 (正弦波,方波,三角波) */

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

#includereg52.h

#include intrins.h

#define uchar unsigned char

#define uint unsigned int

sbit cs=P2^0; //tlc5615片選端口

sbit clk=P2^1; //tlc5615時(shí)鐘線

sbit din=P2^2; //tlc5615傳輸端口

sbit key1=P1^0;

sbit key2=P1^1; //按鍵的單片機(jī)接口

uchar keydat;

uchar flag; //波形發(fā)生終止信號(hào)的標(biāo)志位 一旦被置零立馬停止發(fā)信號(hào)

uchar flagsqu; //方波高低電平控制為(運(yùn)用定時(shí)器1中斷控制)

uchar m,num;

uchar dat=0xff;

uchar code tosin[141]={ //正弦波的編碼

0x00,0x01,0x02,0x02,0x03,0x04,0x05,0x06,0x07,0x08,

0x09,0x0a,0x0b,0x0d,0x0e,0x10,0x11,0x13,0x15,0x16,

0x18,0x1a,0x1c,0x1e,0x20,0x22,0x25,0x27,0x29,0x2b,

0x2e,0x30,0x33,0x35,0x38,0x3a,0x3d,0x40,0x43,0x45,

0x48,0x4c,0x4e,0x51,0x55,0x57,0x5a,0x5d,0x60,0x63,

0x66,0x69,0x6c,0x6f,0x70,0x71,0x72,0x73,0x74,0x75,

0x76,0x77,0x78,0x79,0x7a,0x7b,0x7c,0x7d,0x7e,0x7e,

0x7f,0x80,0x7f,0x7e,0x7e,0x7d,0x7c,0x7b,0x7a,0x79,

0x78,0x77,0x76,0x75,0x74,0x73,0x72,0x6f,0x6c,0x69,

0x66,0x63,0x60,0x5d,0x5a,0x57,0x55,0x51,0x4e,0x4c,

0x48,0x45,0x43,0x40,0x3d,0x3a,0x38,0x35,0x33,0x30,

0x2e,0x2b,0x29,0x27,0x25,0x22,0x20,0x1e,0x1c,0x1a,

0x18,0x16,0x15,0x13,0x11,0x10,0x0e,0x0d,0x0b,0x0a,

0x09,0x08,0x07,0x06,0x05,0x04,0x03,0x02,0x02,0x01,

0x00};

void delay(uchar z) //延時(shí)函數(shù)

{

uchar x,y;

for(x=0;x110;x++)

for(y=z;y0;y--);

}

void prepare() //tlc5615的初始化

{

cs=1;

din=1;

clk=0;

cs=0; //cs的上升沿和下降沿必須在clk為低時(shí)進(jìn)?

}

/* 用中斷來(lái)產(chǎn)生方波

void Squtranslator()

{

TR1=1; //啟動(dòng)定時(shí)器1 控制高低電平的持續(xù)時(shí)間 占空比

do{

do{

_wave=0;

}while((!flagsqu) flag==1);//如果一旦終止信號(hào)的

//產(chǎn)生可以立馬退出循環(huán)

flagsqu=0;

do{

_wave=1;

}while((!flagsqu) flag==1);

flagsqu=0;

}while(flag);

flag=1;

TR1=0;

}

*/

void Squtranslator() //方波函數(shù)

{

uchar j;

uchar dat1=0x7f;

while(flag)

{

do{

prepare();

dat=dat1;

for(j=0;j12;j++)

{

din=(bit)(dat7); //將數(shù)據(jù)的最高位賦給din

clk=1;

dat=dat1; //一位位的傳輸

clk=0;

}

cs=1; //cs的上升沿和下降沿必須在clk為低時(shí)進(jìn)行

delay(200); //使高低電平持續(xù)一段時(shí)間

if(dat1==0)

dat1=0x7f; //完成了0和0x7f之間的替換

else

dat1=0;

}while(flag);

}

}

void Tratranslator() //鋸齒波的發(fā)生函數(shù)

{

uchar j;

uchar dat1=0x7f;

while(flag)

{

do{

prepare();

dat=dat1;

for(j=0;j12;j++)

{

din=(bit)(dat7); //將數(shù)據(jù)的最高位賦給din

clk=1;

dat=dat1; //一位位的傳輸

clk=0;

}

cs=1; //cs的上升沿和下降沿必須在clk為低時(shí)進(jìn)行

delay(2); //稍加延時(shí)

dat1--;

}while(flag dat1); //一旦有終止信號(hào)就可以停止

do{

prepare();

dat=dat1;

for(j=0;j12;j++)

{

din=(bit)(dat7); //將數(shù)據(jù)的最高位賦給din

clk=1;

dat=dat1; //一位位的傳輸

clk=0;

}

cs=1; //cs的上升沿和下降沿必須在clk為低時(shí)進(jìn)行

delay(2); //稍加延時(shí)

dat1++;

}while(flag (!(dat1==0x7f)));

}

}

void Sintranslator(uchar wave[],uchar num )//正弦波的轉(zhuǎn)換函數(shù)

{

uchar i,j;

uchar dat1;

do{

for(i=0;inum;i++)

{

prepare();

dat1=wave[i]; //打開片選 開始工作

for(j=0;j12;j++)

{

din=(bit)(dat17); //將數(shù)據(jù)的最高位賦給din

clk=1;

dat1=dat11; //一位位的傳輸

clk=0;

if(flag==0)break;

}

cs=1; //cs的上升沿和下降沿必須在clk為低時(shí)進(jìn)行

delay(1); //稍加延時(shí)

if(flag==0)break;

}

}while(flag); //等待控制鍵的暫停

}

void keyscan() //切換功能按鍵返回鍵值函數(shù)

{

uchar i;

for(i=0;i4;i++)

{

if(key1==0)

{

delay(10);

if(key1==0)

{

keydat++;

do{}while(!key1); //松手檢測(cè)

if(keydat==4)keydat=1;//加滿回零處理

}

}

}

}

void keycountrl() //切斷輸出控制函數(shù)

{

if(key2==0)

{

delay(10);

if(key2==0)

{

flag=0;

do{}while(!key2); //松手檢測(cè)

}

}

}

void main ()

{

uchar temp;

TMOD=0x01; //確定定時(shí)器的工作方式

TH0=(65536-50000)/256; //給定時(shí)器0賦予初值

TL0=(65536-50000)%256;

EA=1; //開總中斷

ET0=1; //開啟定時(shí)器0中斷

TR0=1;

while(1)

{

do{

switch(keydat)

{

case


分享標(biāo)題:c語(yǔ)言鋸齒波發(fā)生函數(shù) 函數(shù)發(fā)生器鋸齒波
瀏覽地址:http://weahome.cn/article/doohidp.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部