在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

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

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

基于STM32的波形發(fā)生器設計

CHANBAEK ? 來源:單片機資料寶庫 ? 作者: 醉生夢死 ? 2023-12-12 15:16 ? 次閱讀

一、項目介紹

信號發(fā)生器在生產實踐和科技領域中有著廣泛的應用,各種波形曲線均可以用三角函數方程式來表示。能夠產生多種波形,如三角波、鋸齒波、矩形波(含方波)、正弦波的電路被稱為函數信號發(fā)生器。函數信號發(fā)生器在電路實驗和設備檢測中具有十分廣泛的用途,例如在通信、廣播、電視系統(tǒng)中,都需要射頻(高頻)發(fā)射。這里的射頻波就是載波,把音頻(低頻)、視頻信號或脈沖信號運載出去,就需要能夠產生高頻的振蕩器。在工業(yè)、農業(yè)、生物醫(yī)學等領域內,如高頻感應加熱、熔煉、淬火、超聲診斷、核磁共振成像等,都需要功率或大或小、頻率或高或低的振蕩器。函數信號發(fā)生器是各種測試和實驗過程中不可缺少的工具,在通信、測量、雷達、控制、教學等領域應用十分廣泛。不論是在生產、科研還是教學上,信號發(fā)生器都是電子工程師信號仿真實驗的最佳工具。而且,信號發(fā)生器的設計方法多,設計技術也越來越先進, 隨著我國經濟和科技的發(fā)展, 對相應的測試儀器和測試手段也提出了更高的要求,信號發(fā)生器己成為測試儀器中至關重要的一類,因此開發(fā)信號發(fā)生器具有重大意義。

二、主程序流程圖

對系統(tǒng)運行工作流程進行說明后,給出系統(tǒng)主程序流程。

設計思路:首先將所有配置進行初始化,為了消除干擾和提升仿真效率,我們將系統(tǒng)的工作狀態(tài)分為兩種,一種為調整狀態(tài),另一種為波形輸出狀態(tài),使用一個開關的通斷來調整。

系統(tǒng)主程序流程圖如圖4-1所示。

圖片

三、 Proteus軟件仿真調試

正弦波

圖片

方波

圖片

三角波

圖片

鋸齒波

圖片

四、 硬件調試

在實際電路中,我們可以將頻率調整的快一點,但是要注意抗干擾問題,畢竟上面的都是理論值,模擬信號非常容易受到其他因素的干擾。所有配置,都必須跟程序里面完全一致,如果波形失真,或者頻率不對,還應進行相對應的補償。

五、 程序清單

#include "stm32f10x.h"

#include "sys.h"

#include "delay.h"

#include "12864.h"

#include "key4_4.h"

#include "timer.h"

#define KEY0 GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_10)//讀取按鍵0

void Delay_Ms(u16 time);

/*************** 配置Switch用到的I/O口*******************/

void Init_GPIO_Switch(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable, ENABLE);

//關閉jtag,使能SWD,可以用SWD模式調試

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);

// 使能PC端口時鐘

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;

//PC0

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

//IO口速度為50MHz

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;

//設置成輸入

GPIO_Init(GPIOB, &GPIO_InitStructure);

//初始化PC0

}

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

*功能名稱:main

*描述:主程序。

*輸入:無

*輸出:無

*返回:無

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

int main(void)

{

u8 i=0;

RCC_ClocksTypeDef RCC_Clocks; //初始化程序

RCC_Configuration(RCC_PLLMul_4); //8M*4 == 32M

RCC_GetClocksFreq(&RCC_Clocks); //獲取片上時鐘

Init_12864(); //初始化12864液晶

Key_Init();

Init_GPIO_Switch();

Init_GPIO_DAC0832();

Data0=25;

TIM3_Int_Init(50+Data0,320);

//頻率:32000000/ 320 ==100 000 /100 == 1000 /50==20

LCD_P6x8Str(3,16," Sine Wave ");

LCD_P6x8Str(7,6*2,"Frequency: 15 Hz");

while (1)

{

if(KEY0)

{

if(i!=2)

{

__set_PRIMASK(1);

GPIO_ResetBits(GPIOB, ((uint16_t)0xC000));

}

Key_Test();

i=2;

}

else{

if(i!=5)

{

TIM3_Int_Init(50+Data0,320);

__set_PRIMASK(0); //使能TIMx外設

GPIO_ResetBits(GPIOB, ((uint16_t)0xC000));

}

i=5;

}

}

}

/*************** 定時器*******************/

#include "timer.h"

#include "math.h"

#define ADC_CS_WR(a) if (a)

GPIO_SetBits(GPIOB,GPIO_Pin_14);

else

GPIO_ResetBits(GPIOB,GPIO_Pin_14)

#define ADC_CS_WR2(a) if (a)

GPIO_SetBits(GPIOB,GPIO_Pin_15);

else

GPIO_ResetBits(GPIOB,GPIO_Pin_15)

#define PI 3.1415926f //圓周率

u8 mode; //模式:正弦波……

u16 freq; //頻率

u8 time; //計次參數

u8 AM; //調幅

/*************** 配置DAC用到的I/O口*******************/

void Init_GPIO_DAC0832(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE); // 使能PC端口時鐘

GPIO_InitStructure.GPIO_Pin = ((uint16_t)0x03FF); //選擇對應的引腳

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽輸出

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //IO口速度為50MHz

GPIO_Init(GPIOC, &GPIO_InitStructure); //初始化PC端口

GPIO_SetBits(GPIOC, ((uint16_t)0x00FF)); // 高

GPIO_ResetBits(GPIOC, ((uint16_t)0x00FF)); // 低

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE); // 使能PC端口時鐘

GPIO_InitStructure.GPIO_Pin = ((uint16_t)0xC000); //選擇對應的引腳

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽輸出

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //IO口速度為50MHz

GPIO_Init(GPIOB, &GPIO_InitStructure); //初始化PC端口

GPIO_SetBits(GPIOC, ((uint16_t)0xC000)); // 高

GPIO_ResetBits(GPIOC, ((uint16_t)0xC000)); // 低

time=0;

ADC_CS_WR(0);

mode=0; //默認輸出正弦波

freq=100; //默認頻率

AM=255; //最大幅度

}

void DAC_0832_Data(uint8_t Data)

{

GPIO_ResetBits(GPIOC,~Data);

GPIO_SetBits(GPIOC,Data);

}

void sine_wave(u8 location)//輸出正弦波

{

double x=(double)location/50*PI;//把0-100放縮到0-2派(pai,沒有那個符號)

u8 y=(sin(x)*(AM/2)+(AM/2));//算出y,并放縮到0-254(因為ADC范圍0-AM,芯片落后)

DAC_0832_Data(y);

}

void squ_wave(u8 location)//方……

{

if(location<50)

DAC_0832_Data(AM);

else

DAC_0832_Data(0);//這個簡單

}

void tri_wave(u8 location)//三……

{

//為了簡化,在單周期輸出V字形

u8 y;

if(location<50)

y=(50-location)*AM/50;

else

y=(location-50)*AM/50;

DAC_0832_Data(y);

//偶函數,當然說奇函數也沒錯

}

void saw_wave(u8 location)//鋸……

{

DAC_0832_Data(location*AM/100);

//用(100-location)也以變成反向鋸齒

}

//通用定時器中斷初始化

//這里時鐘選擇為APB1的2倍,而APB1為36M

//arr:自動重裝值。

//psc:時鐘預分頻數

//這里使用的是定時器3!

void TIM3_Int_Init(u16 arr,u16 psc)

{

TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;

NVIC_InitTypeDef NVIC_InitStructure;

RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); //時鐘使能

TIM_TimeBaseStructure.TIM_Period = arr; //設置在下一個更新事件裝入活動的自動重裝載寄存器周期的值 計數到5000為500ms

TIM_TimeBaseStructure.TIM_Prescaler =psc; //設置用來作為TIMx時鐘頻率除數的預分頻值 10Khz的計數頻率

TIM_TimeBaseStructure.TIM_ClockDivision = 0; //設置時鐘分割:TDTS = Tck_tim

TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上計數模式

TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根據TIM_TimeBaseInitStruct中指定的參數初始化TIMx的時間基數單位

TIM_ITConfig( //使能或者失能指定的TIM中斷

TIM3, //TIM3

TIM_IT_Update ,

ENABLE //使能

);

// TIM_ITConfig(TIM3, TIM_IT_Update, ENABLE);

NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; //TIM3中斷

NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //先占優(yōu)先級0級

NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //從優(yōu)先級3級

NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能

NVIC_Init(&NVIC_InitStructure); //根據NVIC_InitStruct中指定的參數初始化外設NVIC寄存器

TIM_Cmd(TIM3, ENABLE); //使能TIMx外設

// TIM_Cmd(TIM3, DISABLE);

}

void TIM3_IRQHandler(void) //TIM3中斷

{

switch(mode)

{

case W_SINE:sine_wave((u8)(time*freq/100)%100);break;//計算出波的位置

case W_SQU:squ_wave((u8)((time*freq/100)%100));break;

case W_TRI:tri_wave((u8)((time*freq/100)%100));break;

case W_SAW:saw_wave((u8)((time*freq/100)%100));break;

}

time++;

if(time>=100)//計數100次

time=0;

}

#include "key4_4.h"

#include "delay.h"

#include "sys.h"

#include "12864.h"

#include "timer.h"

//8個引腳 4個為行 4個為列

//行輸出端口定義

#define X1_GPIO_PORT GPIOB

#define X2_GPIO_PORT GPIOB

#define X3_GPIO_PORT GPIOB

#define X4_GPIO_PORT GPIOB

//列輸入端口定義

#define Y1_GPIO_PORT GPIOB

#define Y2_GPIO_PORT GPIOB

#define Y3_GPIO_PORT GPIOB

#define Y4_GPIO_PORT GPIOB

//行輸出引腳定義

#define X1_GPIO_PIN GPIO_Pin_0

#define X2_GPIO_PIN GPIO_Pin_1

#define X3_GPIO_PIN GPIO_Pin_2

#define X4_GPIO_PIN GPIO_Pin_3

//列輸入引腳定義

#define Y1_GPIO_PIN GPIO_Pin_4

#define Y2_GPIO_PIN GPIO_Pin_5

#define Y3_GPIO_PIN GPIO_Pin_6

#define Y4_GPIO_PIN GPIO_Pin_7

//行輸出時鐘定義

#define X1_RCC RCC_APB2Periph_GPIOB

#define X2_RCC RCC_APB2Periph_GPIOB

#define X3_RCC RCC_APB2Periph_GPIOB

#define X4_RCC RCC_APB2Periph_GPIOB

//列輸入時鐘定義

#define Y1_RCC RCC_APB2Periph_GPIOB

#define Y2_RCC RCC_APB2Periph_GPIOB

#define Y3_RCC RCC_APB2Periph_GPIOB

#define Y4_RCC RCC_APB2Periph_GPIOB

//移植代碼只需要修改上面的端口和引腳和時鐘即可,下面的代碼不用修改。

//矩陣鍵盤所用的8個引腳可連續(xù)可不連續(xù),看實際需要和個人愛好自己定義。

unsigned char Y1,Y2,Y3,Y4;

void Key_Init(void)

{

GPIO_InitTypeDef GPIO_InitStructure;

RCC_APB2PeriphClockCmd(X1_RCC|X2_RCC|X3_RCC|X4_RCC|Y1_RCC|Y2_RCC|Y3_RCC|Y4_RCC|RCC_APB2Periph_AFIO, ENABLE);

GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable, ENABLE);

/*******************4行輸出 *****************************/

GPIO_InitStructure.GPIO_Pin = X1_GPIO_PIN ;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;

GPIO_Init(X1_GPIO_PORT, &GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = X2_GPIO_PIN ;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;

GPIO_Init(X2_GPIO_PORT, &GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = X3_GPIO_PIN ;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;

GPIO_Init(X3_GPIO_PORT, &GPIO_InitStructure);

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;

GPIO_InitStructure.GPIO_Pin = X4_GPIO_PIN ;

GPIO_Init(X4_GPIO_PORT, &GPIO_InitStructure);

/******************* 4列輸入 *******************/

GPIO_InitStructure.GPIO_Pin = Y1_GPIO_PIN ;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;

GPIO_Init(Y1_GPIO_PORT, &GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = Y2_GPIO_PIN ;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;

GPIO_Init(Y2_GPIO_PORT, &GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = Y3_GPIO_PIN ;

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;

GPIO_Init(Y3_GPIO_PORT, &GPIO_InitStructure);

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz;

GPIO_InitStructure.GPIO_Pin = Y4_GPIO_PIN;

GPIO_Init(Y4_GPIO_PORT, &GPIO_InitStructure);

}

int Key_Scan(void)

{

uchar KeyVal;

GPIO_SetBits(X1_GPIO_PORT,X1_GPIO_PIN); //先讓X1輸出高

GPIO_SetBits(X2_GPIO_PORT,X2_GPIO_PIN); //先讓X2輸出高

GPIO_SetBits(X3_GPIO_PORT,X3_GPIO_PIN); //先讓X3輸出高

GPIO_SetBits(X4_GPIO_PORT,X4_GPIO_PIN); //先讓X4輸出高

if((GPIO_ReadInputDataBit(Y1_GPIO_PORT,Y1_GPIO_PIN)|GPIO_ReadInputDataBit(Y2_GPIO_PORT,Y2_GPIO_PIN)|GPIO_ReadInputDataBit(Y3_GPIO_PORT,Y3_GPIO_PIN)|GPIO_ReadInputDataBit(Y4_GPIO_PORT,Y4_GPIO_PIN))==0x0000)

return -1; //如果X1到X4全為零則沒有按鍵按下

else

{

Delay_Ms(5); //延時5ms去抖動

if((GPIO_ReadInputDataBit(Y1_GPIO_PORT,Y1_GPIO_PIN)|GPIO_ReadInputDataBit(Y2_GPIO_PORT,Y2_GPIO_PIN)|GPIO_ReadInputDataBit(Y3_GPIO_PORT,Y3_GPIO_PIN)|GPIO_ReadInputDataBit(Y4_GPIO_PORT,Y4_GPIO_PIN))==0x0000)

return -1;

}

GPIO_ResetBits(X1_GPIO_PORT,X1_GPIO_PIN);

GPIO_ResetBits(X2_GPIO_PORT,X2_GPIO_PIN);

GPIO_ResetBits(X3_GPIO_PORT,X3_GPIO_PIN);

GPIO_SetBits(X4_GPIO_PORT,X4_GPIO_PIN);

Y1=GPIO_ReadInputDataBit(Y1_GPIO_PORT,Y1_GPIO_PIN);Y2=GPIO_ReadInputDataBit(Y2_GPIO_PORT,Y2_GPIO_PIN);

Y3=GPIO_ReadInputDataBit(Y3_GPIO_PORT,Y3_GPIO_PIN);Y4=GPIO_ReadInputDataBit(Y4_GPIO_PORT,Y4_GPIO_PIN);

if(Y1==1&&Y2==0&&Y3==0&&Y4==0)

KeyVal='*';

if(Y1==0&&Y2==1&&Y3==0&&Y4==0)

KeyVal=0;

if(Y1==0&&Y2==0&&Y3==0&&Y4==1)

KeyVal='D';

if(Y1==0&&Y2==0&&Y3==1&&Y4==0)

KeyVal='#';

while(((GPIO_ReadInputDataBit(Y1_GPIO_PORT,Y1_GPIO_PIN))|(GPIO_ReadInputDataBit(Y2_GPIO_PORT,Y2_GPIO_PIN))|(GPIO_ReadInputDataBit(Y3_GPIO_PORT,Y3_GPIO_PIN))|(GPIO_ReadInputDataBit(Y4_GPIO_PORT,Y4_GPIO_PIN))) > 0);

//等待按鍵釋放

GPIO_SetBits(X1_GPIO_PORT,X1_GPIO_PIN);

GPIO_ResetBits(X2_GPIO_PORT,X2_GPIO_PIN);

GPIO_ResetBits(X3_GPIO_PORT,X3_GPIO_PIN);

GPIO_ResetBits(X4_GPIO_PORT,X4_GPIO_PIN);

Y1=GPIO_ReadInputDataBit(Y1_GPIO_PORT,Y1_GPIO_PIN);Y2=GPIO_ReadInputDataBit(Y2_GPIO_PORT,Y2_GPIO_PIN);

Y3=GPIO_ReadInputDataBit(Y3_GPIO_PORT,Y3_GPIO_PIN);Y4=GPIO_ReadInputDataBit(Y4_GPIO_PORT,Y4_GPIO_PIN);

if(Y1==1&&Y2==0&&Y3==0&&Y4==0)

KeyVal=1;

if(Y1==0&&Y2==1&&Y3==0&&Y4==0)

KeyVal=2;

if(Y1==0&&Y2==0&&Y3==1&&Y4==0)

KeyVal=3;

if(Y1==0&&Y2==0&&Y3==0&&Y4==1)

KeyVal='A';

while(((GPIO_ReadInputDataBit(Y1_GPIO_PORT,Y1_GPIO_PIN))|(GPIO_ReadInputDataBit(Y2_GPIO_PORT,Y2_GPIO_PIN))|(GPIO_ReadInputDataBit(Y3_GPIO_PORT,Y3_GPIO_PIN))|(GPIO_ReadInputDataBit(Y4_GPIO_PORT,Y4_GPIO_PIN))) > 0);

GPIO_ResetBits(X1_GPIO_PORT,X1_GPIO_PIN);

GPIO_SetBits(X2_GPIO_PORT,X2_GPIO_PIN);

GPIO_ResetBits(X3_GPIO_PORT,X3_GPIO_PIN);

GPIO_ResetBits(X4_GPIO_PORT,X4_GPIO_PIN);

Y1=GPIO_ReadInputDataBit(Y1_GPIO_PORT,Y1_GPIO_PIN);Y2=GPIO_ReadInputDataBit(Y2_GPIO_PORT,Y2_GPIO_PIN);

Y3=GPIO_ReadInputDataBit(Y3_GPIO_PORT,Y3_GPIO_PIN);Y4=GPIO_ReadInputDataBit(Y4_GPIO_PORT,Y4_GPIO_PIN);

if(Y1==1&&Y2==0&&Y3==0&&Y4==0)

KeyVal=4;

if(Y1==0&&Y2==1&&Y3==0&&Y4==0)

KeyVal=5;

if(Y1==0&&Y2==0&&Y3==1&&Y4==0)

KeyVal=6;

if(Y1==0&&Y2==0&&Y3==0&&Y4==1)

KeyVal='B';

while(((GPIO_ReadInputDataBit(Y1_GPIO_PORT,Y1_GPIO_PIN))|(GPIO_ReadInputDataBit(Y2_GPIO_PORT,Y2_GPIO_PIN))|(GPIO_ReadInputDataBit(Y3_GPIO_PORT,Y3_GPIO_PIN))|(GPIO_ReadInputDataBit(Y4_GPIO_PORT,Y4_GPIO_PIN))) > 0);

GPIO_ResetBits(X1_GPIO_PORT,X1_GPIO_PIN);

GPIO_ResetBits(X2_GPIO_PORT,X2_GPIO_PIN);

GPIO_SetBits(X3_GPIO_PORT,X3_GPIO_PIN);

GPIO_ResetBits(X4_GPIO_PORT,X4_GPIO_PIN);

Y1=GPIO_ReadInputDataBit(Y1_GPIO_PORT,Y1_GPIO_PIN);Y2=GPIO_ReadInputDataBit(Y2_GPIO_PORT,Y2_GPIO_PIN);

Y3=GPIO_ReadInputDataBit(Y3_GPIO_PORT,Y3_GPIO_PIN);Y4=GPIO_ReadInputDataBit(Y4_GPIO_PORT,Y4_GPIO_PIN);

if(Y1==1&&Y2==0&&Y3==0&&Y4==0)

KeyVal=7;

if(Y1==0&&Y2==1&&Y3==0&&Y4==0)

KeyVal=8;

if(Y1==0&&Y2==0&&Y3==1&&Y4==0)

KeyVal=9;

if(Y1==0&&Y2==0&&Y3==0&&Y4==1)

KeyVal='C';

while(((GPIO_ReadInputDataBit(Y1_GPIO_PORT,Y1_GPIO_PIN))|(GPIO_ReadInputDataBit(Y2_GPIO_PORT,Y2_GPIO_PIN))|(GPIO_ReadInputDataBit(Y3_GPIO_PORT,Y3_GPIO_PIN))|(GPIO_ReadInputDataBit(Y4_GPIO_PORT,Y4_GPIO_PIN))) > 0);

return KeyVal;

}

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

按鍵表盤為: 1 2 3 A

4 5 6 B

7 8 9 C

  • 0 # D

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

u8 ee,Data0=0;

void Key_Test(void)

{

int num;

char Freq[3]={'?'};

num = Key_Scan();

if(ee!=num)

switch(num)

{

case 0: LCD_P6x8Str(1,1,"0"); break;

case 1: LCD_P6x8Str(1,1,"1"); break;

case 2: LCD_P6x8Str(1,1,"2"); break;

case 3: LCD_P6x8Str(1,1,"3"); break;

case 4: LCD_P6x8Str(1,1,"4"); break;

case 5: LCD_P6x8Str(1,1,"5"); break;

case 6: LCD_P6x8Str(1,1,"6"); break;

case 7: LCD_P6x8Str(1,1,"7"); break;

case 8: LCD_P6x8Str(1,1,"8"); break;

case 9: LCD_P6x8Str(1,1,"9"); break;//

case'A':LCD_P6x8Str(1,1,"A");

mode=0;LCD_P6x8Str(3,16," Sine Wave "); break;

case'B':LCD_P6x8Str(1,1,"B");

mode=1;LCD_P6x8Str(3,16," Square wave "); break;

case'C':LCD_P6x8Str(1,1,"C"); mode=2;LCD_P6x8Str(3,16,"Ttiangular wave"); break;

case 'D': LCD_P6x8Str(1,1,"D");

mode=3;LCD_P6x8Str(3,16," Sawtooth Wave ");

break;

case '#': LCD_P6x8Str(1,1,"#");

if(Data0>=5)Data0-=5;

Freq[0]=(20-Data0/5)/10+'0';

Freq[1]=(20-Data0/5)%10+'0';

LCD_P6x8Str(7,6*13,Freq);

break;

case ' ': LCD_P6x8Str(1,1," ");

if(Data0<50)Data0+=5; Freq[0]=(20-Data0/5)/10+'0';

Freq[1]=(20-Data0/5)%10+'0';

LCD_P6x8Str(7,6*13,Freq);

break;

}

ee=num;

}

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

12864

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

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

//寫數據地址

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

void write_12864com(uint8_t com)

{

uint8_t temp = 0x01;

uint8_t k[8] = {0};

uint8_t i;

Check_status();

for(i=0; i<8; i++)

{

if(com & temp)

k[i] = 1;

else

k[i] = 0;

temp=temp << 1;

}

temp = 0x01;

RS=0;

RW=0;

D0=k[0];

D1=k[1];

D2=k[2];

D3=k[3];

D4=k[4];

D5=k[5];

D6=k[6];

D7=k[7];

En=1;

Delay_Ms(1);

En=0;

Delay_Ms(1);

}

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

//寫數據

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

void write_12864data(uint8_t dat)

{

uint8_t temp = 0x01;

uint8_t k[8] = {0};

uint8_t i;

Check_status();//延時檢查很重要,在連續(xù)刷新的時候

for(i=0; i<8; i++)

{

if(dat & temp)

k[i] = 1;

else

k[i] = 0;

temp=temp << 1;

}

temp = 0x01;

RS=1;

RW=0;

D0=k[0];

D1=k[1];

D2=k[2];

D3=k[3];

D4=k[4];

D5=k[5];

D6=k[6];

D7=k[7];

En=1;//在使能信號的下降沿寫入了數據

Delay_Ms(1);

En=0;

Delay_Ms(1);

}

uint8_t read_12864data()

{

//這個函數的代碼測試未通過

uint8_t temp = 0;

uint8_t k[8] = {0};

uint8_t i;

Check_status();

RS=1;

RW=1;

En=1;//使能端高電平讀取數據

Delay_Ms(1);

LCD12864_DataRead_PortInit();//端口方向改為輸入

k[0]=DI0;

k[1]=DI1;

k[2]=DI2;

k[3]=DI3;

k[4]=DI4;

k[5]=DI5;

k[6]=DI6;

k[7]=DI7;

LCD12864_DataWrite_PortInit();//端口方向改為輸出

for (i=0; i<8; i++)

{

temp |= ((k[i]&0x01)<

}

return temp;

}

void LCD_Set_Pos(uint8_t page,uint8_t x)//LCD當前光標設置,先選頁,后選橫向起始坐標

{

uint8_t Column=0;

(x>127)?(x=127):(x=x);

(x<1)?(x=0):(x=x);

if (page>8)

{

page=8;

}

else if (page<1)

{

page=1;

}

page=page-1;

if (x<=63)

{ LCDSelectScreen(LeftScreen);

LCDSetPage(page);

Column=x;

LCDSetColumn(Column);

}

else if (x>63)

{ LCDSelectScreen(RightScreen);

LCDSetPage(page);

Column = x-64;

LCDSetColumn(Column);

// if (Column == 0)

// { //如果第一次進入右半屏,需要空寫一次,在設置一次,方可

// write_12864data(0x00);

// }

LCDSelectScreen(RightScreen);//上面的判斷不對,應該是寫兩次命令就會解決右半屏首字下沉那個問題

LCDSetPage(page);

Column = x-64;

LCDSetColumn(Column);

}

}

void LCD_P6x8Str(uint8_t x, uint8_t y, char ch[])

{

uint8_t c = 0, i = 0, j = 0;

while (ch[j] != '?')

{

c = ch[j]-32;

for (i = 0; i < 6; i++)

{ LCD_Set_Pos(x, y);//因為這個AMPIRE 分為左右半屏,所以光標設置必須寫在這里,確保每個字符是完整的

write_12864data(F6x8[c][i]);

y = y+1;

}

j++;

}

}

void LCD_DispACat(unsigned char x, unsigned char y)

{

unsigned char i = 0, j = 0;

if (x > 4)

{

x = 4;

y++;

}

x=x-1;

for (j = 0; j < 4; j++)

{

LCD_Set_Pos(x, y);

x++;

for (i = 0; i < 32; i++)

write_12864data(LCD_cat[i + j * 32]);

}

}

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

//12864初始化

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

void Init_12864()

{

//端口初始化

RCC->APB2ENR|=1<<2; //使能PORTA時鐘

GPIOA->CRH&=0X00000000;

GPIOA->CRH|=0X44333333;//PA8~13推挽輸出,PA14,15浮空輸入,因為這個型號的LCD要檢測內部是否忙

GPIOA->ODR|=0X00FFFFFF;//PA8~13輸出高

LCD12864_DataWrite_PortInit();

Reset=0;

Delay_Ms(1);

Reset=1;

Check_status();

LCDSelectScreen(AllScreen);//屏幕選擇全屏

LCDSetOnOff(Off);

LCDSelectScreen(AllScreen);//屏幕選擇全屏

LCDSetOnOff(On);

LCDSelectScreen(AllScreen);//屏幕選擇全屏

LCDClearScreen(AllScreen);//清屏

LCDSetLine(0);

}

void LCD12864_DataWrite_PortInit()

{

RCC->APB2ENR|=1<<2;

GPIOA->CRL&=0X00000000; //PORTA低八位為數據位,推挽輸出,寫數據用

GPIOA->CRL|=0X33333333;

GPIOA->ODR|=0XFFFFFFFF;

}

void LCD12864_DataRead_PortInit()

{

RCC->APB2ENR|=1<<2;

GPIOA->CRL&=0X00000000; //PORTA低八位為數據位,浮空輸入,讀數據用

GPIOA->CRL|=0X44444444;

//GPIOA->ODR|=0XFFFFFFFF;

}

void LCDSelectScreen(ScreenTypedef screen)

{

switch(screen)

{

case AllScreen:

CS1=0;

CS2=0;

break;

case LeftScreen:

CS1=0;

CS2=1;

break;

case RightScreen:

CS1=1;

CS2=0;

break;

default :

break;

}

}

void LCDSetOnOff(LCDSetOnOffTypedef Status)

{

switch (Status)

{

case On:

write_12864com(0x3e+1);

break;

case Off:

write_12864com(0x3e);

break;

default:

break;

}

}

void LCDClearScreen(ScreenTypedef screen)

{

uint8_t i,j;

LCDSelectScreen(screen);

for (i=0; i<=7; i++)

{

LCDSetPage(i);

LCDSetColumn(0);

for(j=0; j<=63; j++)

{

write_12864data(0x00);//寫完后地址自動加一

}

}

}

void LCDSetPage(uint8_t Page)

{

Page=0xb8|Page;

write_12864com(Page);

}

void LCDSetLine(uint8_t Line)

{

Line=0xc0|Line;

write_12864com(Line);

}

void LCDSetColumn(uint8_t Colnum)

{

Colnum=Colnum&0x3f;

Colnum=Colnum|0x40;

write_12864com(Colnum);

}

void Check_status()

{

D0=0;

D1=0;

D2=0;

D3=0;

D4=0;

D5=0;

D6=0;

D7=0;

RS=0;

RW=1;

//En=1;

// while(CheckPin == 0);//不啟用忙檢測

Delay_Ms(2);

//En=0;

}

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯系本站處理。 舉報投訴
  • 正弦波
    +關注

    關注

    11

    文章

    647

    瀏覽量

    55506
  • STM32
    +關注

    關注

    2270

    文章

    10914

    瀏覽量

    356712
  • 信號發(fā)生器

    關注

    28

    文章

    1478

    瀏覽量

    108855
  • 波形發(fā)生器

    關注

    3

    文章

    293

    瀏覽量

    31402
收藏 人收藏

    評論

    相關推薦

    基于stm32波形發(fā)生器設計

    #include “dac.h”//DAC 通道 1 輸出初始化void Dac1_Init(void){GPIO_InitTypeDef GPIO_InitStructure;DAC_InitTypeDef DAC_InitType;RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE ); //①使能 PA 時鐘RCC_APB1PeriphClockCmd(RCC_APB1Periph_DAC, ENABLE ); //②使能DAC時鐘G
    發(fā)表于 08-16 08:39

    如何去實現一種基于stm32波形發(fā)生器

    如何去實現一種基于stm32波形發(fā)生器呢?其代碼程序該怎樣去實現呢?
    發(fā)表于 11-17 06:19

    怎么實現基于STM32波形發(fā)生器的設計?

    怎么實現基于STM32波形發(fā)生器的設計?
    發(fā)表于 11-26 06:16

    程序波形發(fā)生器電路

    程序波形發(fā)生器電路 程序波形發(fā)生器
    發(fā)表于 02-09 16:07 ?995次閱讀
    程序<b class='flag-5'>波形</b><b class='flag-5'>發(fā)生器</b>電路

    SPWM波形發(fā)生器設計與應用

    SPWM波形發(fā)生器設計與應用
    發(fā)表于 07-24 16:11 ?6113次閱讀
    SPWM<b class='flag-5'>波形</b><b class='flag-5'>發(fā)生器</b>設計與應用

    波形發(fā)生器設計簡述

    在工作中,我們常常會用到波形發(fā)生器,它是使用頻度很高的電子儀器。現在的波形發(fā)生器都采用單片機來構成。單片機波形
    發(fā)表于 03-28 15:43 ?0次下載

    波形發(fā)生器

    波形發(fā)生器,可以生成4中波形。正玄波,矩形波,三角波等。
    發(fā)表于 05-13 10:17 ?30次下載

    波形發(fā)生器的設計

    波形發(fā)生器的設計
    發(fā)表于 12-17 21:49 ?30次下載

    波形發(fā)生器軟件

    0-20波形發(fā)生器
    發(fā)表于 12-28 11:07 ?4次下載

    簡易波形發(fā)生器

    簡易波形發(fā)生器
    發(fā)表于 01-05 14:34 ?29次下載

    波形發(fā)生器

    多種波形發(fā)生器課程設計。
    發(fā)表于 02-08 02:37 ?180次下載

    波形發(fā)生器應用的測量

    波形發(fā)生器應用的測量
    發(fā)表于 03-28 10:24 ?1次下載

    關于波形發(fā)生器的設計

    波形發(fā)生器
    發(fā)表于 07-03 15:31 ?120次下載

    iboard教程之基于STM32波形發(fā)生器的軟件分析

    發(fā)表于 10-29 09:22 ?0次下載
    iboard教程之基于<b class='flag-5'>STM32</b>的<b class='flag-5'>波形</b><b class='flag-5'>發(fā)生器</b>的軟件分析

    關于波形發(fā)生器,你知道多少?

    信號發(fā)生器系列波形發(fā)生器你知道多少?虹科信號發(fā)生器系列波形發(fā)生器
    的頭像 發(fā)表于 08-05 08:07 ?3140次閱讀
    關于<b class='flag-5'>波形</b><b class='flag-5'>發(fā)生器</b>,你知道多少?
    主站蜘蛛池模板: 国产在线高清精品二区色五郎| 99久久精品费精品国产一区二| 你懂的免费在线| 色妇影院| 色人在线| 欧美成人aaa大片| 精品手机在线| 国产高清免费| 午夜影院日韩| 国内自拍2021| 88xx成人永久免费观看| 黄.www| 无遮挡一级毛片| 人人草在线| 国产好深好硬好爽我还要视频| 欧洲精品码一区二区三区免费看| 永久黄网站色视频免费观看| 在线久综合色手机在线播放| 性xxxxbbbb在线| 欧美一级高清片在线| 精品美女在线| 国产免费卡1卡2卡| 天天爽夜夜爽夜夜爽| 国产精品国产午夜免费福利看| 男人午夜影院| 又黄又免费的网站| 欧美丝袜一区| 天堂网最新| 亚洲天天更新| 欧美成人精品一区二三区在线观看| 无遮挡很爽很污很黄在线网站| 在线观看免费国产| 日韩成人一级| 日韩亚洲欧美日本精品va| 久操视频在线| 亚洲卡一卡2卡三卡4卡国色| 性欧美暴力猛交69hd| 男人的天堂免费网站| 国产黄色精品| 久久riav国产精品| 色黄网|