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

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

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

3天內不再提示

STM32F103雙重ADC同步規則模式采集實驗

王琪 ? 來源:pgph ? 作者:pgph ? 2022-02-24 10:47 ? 次閱讀

雙重 ADC 同步規則模式采集實驗與多路LCD 波形示波器制作顯示,本文展示了STM32 AD 雙重 ADC 同步規則模式采集實驗。

內容涉及 :

STM32 AD 雙重 ADC 同步規則

AD多通道DMA采集與存儲調用

AD采樣點的構造體封裝

AD 的處理以及 LCD波形輸出 模仿示波器的原理

LCD觸摸畫板的控制

SRAM 內存擴展管理

FatFs 文件系統移植

SPI函數移植過程

SPI字節數據模擬輸出獨寫 緩存讀寫

USART串口的識別

IO口輸入輸出

按鍵的外部中斷處理

32位數據通訊,字符串通訊,單字符通訊

一:編程要點

初始 ADC 用到的 GPIO;

初始化 ADC GPIO;

初始化 DMA 配置;

初始化 ADC 參數;

讀取 ADC 采集的數據,并打印出來校正;

設置 ADC 的工作參數并初始化;

設置 ADC 工作時鐘;

設置 ADC 轉換通道順序及采樣時間;

配置 DMA 工作參數;;

使能 ADC 7) 讀取 ADC 采集的數據。

同步規則模式是 ADC1 和 ADC2 同時轉換一個規則通道組,ADC1 是主,ADC2 是從。

ADC1 轉換的結果放在 ADC1_DR 的低 16位,ADC2 轉換的結果放在 ADC1_DR 的高十六位。

并且必須開啟 DMA 功能。外部觸發來自 ADC1 的規則組多路開關(由 ADC1_CR2 寄存器的 EXTSEL[2:0]選擇)。

它同時給 ADC2 提供同步觸發。為了簡單起見,ADC1 我們選擇軟件觸發,ADC2 必須選擇外部觸發。

這個外部觸發來自于 ADC1 的規則組多路開關。

二:ADC 的工作具體如下

AD轉換包括采樣階段和轉換階段,在采樣階段才對通道數據進行采集;

而在轉換階段只是將采集到的數據進行轉換為數字量輸出,此刻通道數據變化不會改變轉換結果。

獨立模式的 ADC 采集需要在一個通道采集并且轉換完成后才會進行下一個通道的采集。

而雙重 ADC 的機制就是使用兩個 ADC 同時采樣一個或者多個通道。

雙重 ADC 模式較獨立模式一個最大的優勢就是提高了采樣率,彌補了單個 ADC 采樣不夠快的缺點。

啟用雙 ADC模式的時候,通過配置 ADC_CR1寄存器的 DUALMOD[3:0]位,可以有幾種不同的模式,具體見表格 31-1

模式 簡要說明
同步注入模式 ADC1和ADC2同時轉換一個注入通道組,其中ADC1為主,ADC2為從。轉換的數據存儲在每個 ADC 接口的 ADC_JDRx 寄存器中。
同步規則模式 ADC1 和 ADC2 同時轉換一個規則通道組,其中 ADC1 為主,ADC2 為從。
ADC1 轉換的結果放在 ADC1_DR 的低 16 位,ADC2 轉換的結果放在 ADC1_DR 的高十六位。
快速交叉模式 ADC1 和 ADC2 交替采集一個規則通道組(通常為一個通道)。當ADC2 觸發之后,ADC1 需要等待 7 個 ADCCLK 之后才能觸發。
慢速交叉模式 ADC1 和 ADC2 交替采集一個規則通道組(只能為一個通道)。當ADC2 觸發之后,ADC1 需要等待 14 個 ADCCLK 之后才能觸發。

三:代碼分析

1:ADC_book.h

#ifndef      __ADC_BOOK_H
#define	     __ADC_BOOK_H
#include "stm32f10x.h"

// ADC GPIO宏定義
// 注意:用作ADC采集的IO必須沒有復用,否則采集電壓會有影響// ADC 編號選擇
// 可以是 ADC1/2,如果使用ADC3,中斷相關的要改成ADC3的
#define    ADC_GPIO_APBxClock_FUN        RCC_APB2PeriphClockCmd
#define    ADC_GPIO_CLK                  RCC_APB2Periph_GPIOC  
#define    ADC_PORT                      GPIOC

// ADC 編號選擇
// 可以是 ADC1/2,如果使用ADC3,中斷相關的要改成ADC3的
#define   ADC_APBxClock_FUN             RCC_APB2PeriphClockCmd
#define   ADC_X                         ADC1 //ADC2
#define   ADC_Y                         ADC2 //ADC2
#define   ADC_CLKX                      RCC_APB2Periph_ADC1
#define   ADC_CLKY                      RCC_APB2Periph_ADC2

//-------------------------------- ADC DMA 配置 ------------------------
         
 // ADC1 對應 DMA1 通道 1,ADC3 對應 DMA2 通道 5,ADC2 沒有 DMA 功能
#define 	ADC_DMA_CLK  									  RCC_AHBPeriph_DMA1
#define 	ADC_DMA_CHANEL 									DMA1_Channel1
#endif    

//ADC   中斷相關宏定義
#define    ADC_IRQ                       ADC1_2_IRQn
#define    ADC_IRQHandler                ADC1_2_IRQHandler

//-------------------------------- ADC配置 ------------------------------
 
// 雙通道ADC同步設計
#define  	 __ADC_RegSimult_Mode__ 						//使能標志位
#ifdef    __ADC_RegSimult_Mode__ 
#define    NOFCHANEL										 2  //轉換通道的個數
#define    _DMA_BufferSize               1
 
  	#define    ADC_PIN1                      GPIO_Pin_1
	#define    ADC_CHANNEL1                  ADC_Channel_11
	#define    ADC_PIN4                      GPIO_Pin_4
	#define    ADC_CHANNEL4                  ADC_Channel_14

#endif 

 
typedef union {
  struct{
    unsigned char BIT0:1;unsigned char BIT1:1;unsigned char BIT2:1;unsigned char BIT3:1;
    unsigned char BIT4:1;unsigned char BIT5:1;unsigned char BIT6:1;unsigned char BIT7:1;
    //unsigned char BIT8:1;unsigned char BIT9:1;unsigned char BIT10:1;unsigned char BIT11:1;
    //unsigned char BIT12:1;unsigned char BIT13:1;unsigned char BIT14:1;unsigned char BIT15:1;
  }DATA_BIT;
  uint8_t DATA_BYTE;
}Per_adc_type;

extern volatile  Per_adc_type  adc_flag;
  #define badc_10ms         adc_flag.DATA_BIT.BIT0
 
extern volatile  uint32_t   Count_Adc_flag;


//----------- 雙通道ADC同步設計-------	
#ifdef  	 __ADC_RegSimult_Mode__
	extern volatile  uint32_t   ADC_RegSimult_ConvertedValue[NOFCHANEL];
	extern volatile  float 			ADC_RegSimult_ConvertedValueLocal[NOFCHANEL];
#endif

void ADCx_Init(void);
void ADC_get_value(void);

#endif 

2:ADC_book.c

使用到 GPIO 時候都必須開啟對應的 GPIO 時鐘,GPIO 用于 AD 轉換功能必須配置為模擬輸入模式。

ADCx_Mode_Config()與獨立模式多通道配置基本一樣,只是有幾點需要注意:
ADC 工作模式要設置為同步規則模式;兩個 ADC 的通道的采樣時間需要一致;
ADC1設置為軟件觸發;ADC2 設置為外部觸發。其他的基本一樣,看代碼注釋理解即可。

#include "ADC_book.h"
#include "XPT2046_LCD_GridDiagram_book.h"

volatile  Per_adc_type  adc_flag;		  
volatile  uint32_t 			Count_Adc_flag;

#ifdef  	 __ADC_RegSimult_Mode__ 
__IO 		uint32_t 		ADC_RegSimult_ConvertedValue[NOFCHANEL]={0,0};
__IO 		uint32_t 		ADC_RegSimult_ConvertedValue_Show[NOFCHANEL]={0,0};
__IO 		float 		  ADC_RegSimult_ConvertedValueLocal[NOFCHANEL];
#endif


/**
  * @brief  ADC GPIO 初始化函數
  * @param  無
  * @retval 無
  */
static void ADCx_GPIO_Config(void){
	GPIO_InitTypeDef GPIO_InitStructure;
	// 打開ADC IO 端口時鐘
	ADC_GPIO_APBxClock_FUN (ADC_GPIO_CLK , ENABLE); 
	//配置 ADC IO 端口的模式   
	//-----------雙通道ADC同步設計-------------
	#ifdef  	 __ADC_RegSimult_Mode__
	GPIO_InitStructure.GPIO_Pin = ADC_PIN1|
	ADC_PIN4; 
	#endif 
 
	// 必須為模擬輸入
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; 
	//初始化 ADDC_IO
	GPIO_Init(ADC_PORT , &GPIO_InitStructure );
	
}
	
/**
  * @brief  配置ADC DMA 工作模式
  * @param  
  * @retval 
  */
static void ADCx_DMA_Config(void){
  DMA_InitTypeDef  DMA_InitStructure; 
  //打開DMA時鐘
	RCC_AHBPeriphClockCmd(ADC_DMA_CLK , ENABLE);
	//復位DMA 控制器
  DMA_DeInit(ADC_DMA_CHANEL);
	//配置DMA初始化結構體
  // 外設基址為:ADC 數據寄存器地址
	DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)(&(ADC_X->DR));
	
  // 存儲器地址,實際上就是一個內部SRAM的變量
	#ifdef  	 __ADC_RegSimult_Mode__
	//----------- 雙通道ADC同步設計-------	
	DMA_InitStructure.DMA_MemoryBaseAddr = (u32)ADC_RegSimult_ConvertedValue;
	#endif 
  // 數據源來自外設
  DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
	// 緩沖區大小為1,緩沖區的大小應該等于存儲器的大小 
  DMA_InitStructure.DMA_BufferSize = _DMA_BufferSize ;
	// 外設寄存器只有一個,地址不用遞增
	DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;

  
	#ifdef  	 __ADC_RegSimult_Mode__
	// 外設數據大小為半字,即兩個字節
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;
	// 存儲器數據大小也為半字,跟外設數據大小相同
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;																							 
	#else
	// 外設數據大小為半字,即兩個字節
	DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
	// 存儲器數據大小也為半字,跟外設數據大小相同
	DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
	#endif

	// 循環傳輸模式
	DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
	// DMA 傳輸通道優先級為高,當使用一個DMA通道時,優先級設置不影響
	DMA_InitStructure.DMA_Priority = DMA_Priority_High;
	// 禁止存儲器到存儲器模式,因為是從外設到存儲器
	DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
	// 初始化DMA
	DMA_Init(ADC_DMA_CHANEL,&DMA_InitStructure);
	// 使能 DMA 通道
	DMA_Cmd(ADC_DMA_CHANEL , ENABLE);
}
	
/**
  * @brief  配置ADC 工作模式
  * @param  
  * @retval 
  */
static void ADCx_Mode_Config(void){
	ADC_InitTypeDef ADC_InitStructure;
  //----------- 雙通道ADC同步設計-------	
  #ifdef  	 __ADC_RegSimult_Mode__
	//打開ADC時鐘 
	ADC_APBxClock_FUN (ADC_CLKX , ENABLE);
	ADC_APBxClock_FUN (ADC_CLKY , ENABLE);
	#else
	ADC_APBxClock_FUN (ADC_CLKX , ENABLE);
  #endif 
  
	//--------ADC 模式配置  -----
	//----------- 雙通道ADC同步設計-------	
  #ifdef  	 __ADC_RegSimult_Mode__
  // 雙 ADC 的規則同步
  ADC_InitStructure.ADC_Mode = ADC_Mode_RegSimult;
  // 掃描模式
	ADC_InitStructure.ADC_ScanConvMode = ENABLE; 
  #endif
    
	//連續轉換模式
	ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
	// 不用外部觸發轉換 ,軟件開啟即可 
	ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
	//轉換結果右側對齊 
	ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;

  //----------- 雙通道ADC同步設計-------	
  #ifdef  	 __ADC_RegSimult_Mode__
	ADC_InitStructure.ADC_NbrOfChannel = NOFCHANEL/2; 
  //初始化ADC 
	ADC_Init(ADC_X , &ADC_InitStructure);
  ADC_Init(ADC_Y , &ADC_InitStructure);
	#endif
  
	//配置ADC時鐘為PCLCK2的8分頻 為9HZ
	RCC_ADCCLKConfig(RCC_PCLK2_Div8);
	//配置ADC轉換通道轉換順序和采樣時間
	
 
  //----------- 雙通道ADC同步設計-------	
  #ifdef  	 __ADC_RegSimult_Mode__
	ADC_RegularChannelConfig(ADC_X , ADC_CHANNEL1 , 1 , ADC_SampleTime_239Cycles5);
	ADC_RegularChannelConfig(ADC_Y , ADC_CHANNEL4 , 1 , ADC_SampleTime_239Cycles5);
	#endif
	//-----------AD通道采集------- 
  //-----------DMA模式----------
	#ifdef 	__ADC_DMA_Mode__   
		#ifdef  	 __ADC_RegSimult_Mode__
		// 只需要使能ADC1 DMA 請求
    //ADC1 我們選擇軟件觸發,ADC2 必須選
		//擇外部觸發,這個外部觸發來自于 ADC1 的規則組多路開關。
		ADC_DMACmd(ADC_X, ENABLE);
     /* 使能 ADCx_2 的外部觸發轉換 */
    ADC_ExternalTrigConvCmd(ADC_Y, ENABLE);
		#else
		// 使能ADC DMA 請求
		ADC_DMACmd(ADC_X, ENABLE);
		#endif 
	#else
	//ADC 轉換結束產生中斷,在中斷服務程序中讀取轉換值
	ADC_ITConfig(ADC_X , ADC_IT_EOC , ENABLE);
	// 使能ADC DMA 請求
	ADC_DMACmd(ADC_X, ENABLE);
	#endif 
	//-----------DMA模式----------
  
	/* ----------------ADC 校準--------------------- */
  //----------- 雙通道ADC同步設計-------	
  #ifdef  	 __ADC_RegSimult_Mode__
	//開啟ADC 并開始轉換
	ADC_Cmd(ADC_X , ENABLE);
	// 初始化ADC 校準寄存器 
	ADC_ResetCalibration(ADC_X);
	// 等待校準寄存器初始化完成
	while(ADC_GetResetCalibrationStatus(ADC_X));
	//ADC_采集校準 
	ADC_StartCalibration(ADC_X);
	//等待校準完成
	while(ADC_GetCalibrationStatus(ADC_X));
  

   //開啟ADC 并開始轉換
	ADC_Cmd(ADC_Y , ENABLE);
	// 初始化ADC 校準寄存器 
	ADC_ResetCalibration(ADC_Y);
	// 等待校準寄存器初始化完成
	while(ADC_GetResetCalibrationStatus(ADC_Y));
	//ADC_采集校準 
	ADC_StartCalibration(ADC_Y);
	//等待校準完成
	while(ADC_GetCalibrationStatus(ADC_Y));
   
	//由于沒有采用外部觸發 ,所以使用軟件ADC轉換
    //ADC 工作模式要設置為同步規則模式;兩個 ADC 的通道的采樣時間需要一致;
    //ADC1設置為軟件觸發;ADC2 設置為外部觸發 
	ADC_SoftwareStartConvCmd(ADC_X , ENABLE);
  
	
}
 

static void ADC_NVIC_Config(void){
	NVIC_InitTypeDef  NVIC_InitStructure;
	//優先級分組
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
	//配置中斷優先級
	NVIC_InitStructure.NVIC_IRQChannel = ADC_IRQ;
	NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
	NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
	NVIC_Init(&NVIC_InitStructure);
}


void ADC_IRQHandler(void){
	if( ADC_GetITStatus(ADC_X  , ADC_IT_EOC )==SET ){
		//讀取ADC的轉換值
		 //--------DMA模式-----
		#ifdef 	__ADC_DMA_Mode__ 
		#else
		ADC_ConvertedValue = ADC_GetConversionValue(ADC_X);		
		#endif 
	//--------DMA模式-----
		ADC_ClearITPendingBit(ADC_X , ADC_IT_EOC);
	}
  if( ADC_GetITStatus(ADC_Y  , ADC_IT_EOC )==SET ){
		//讀取ADC的轉換值
		 //--------DMA模式-----
		#ifdef 	__ADC_DMA_Mode__ 
		#else
		ADC_ConvertedValue = ADC_GetConversionValue(ADC_Y);		
		#endif 
	//--------DMA模式-----
	  ADC_ClearITPendingBit(ADC_Y , ADC_IT_EOC);	
	} 
  
}
 
/**********************END OF FILE**********************/

/**
  * @brief  ADC初始化
  * @param  無
  * @retval 無
  */
 void ADCx_Init(void){
	 //--------DMA模式-----
	#ifdef 	__ADC_DMA_Mode__ 
	ADCx_GPIO_Config();
  ADCx_DMA_Config();
	ADCx_Mode_Config();
	#else
	ADCx_GPIO_Config();
	ADCx_Mode_Config();
	ADC_NVIC_Config();  // DMA 模式不需要對中斷進行處理	
	#endif 
	//--------DMA模式-----
	 
}


void ADC_get_value(void){
  uint16_t temp0=0 ,temp1=0;
  if(badc_10ms==0){return;}
  badc_10ms = 0;

  //-----------AD通道采集-------	
  #ifdef 		 __ADC_RegSimult_Mode__ 
	// 取出 ADC1 數據寄存器的高 16 位,這個是 ADC2 的轉換數據
	temp0 = (ADC_RegSimult_ConvertedValue[0]&0XFFFF0000) >> 16;
	// 取出 ADC1 數據寄存器的低 16 位,這個是 ADC1 的轉換數據
	temp1 = (ADC_RegSimult_ConvertedValue[0]&0XFFFF);
	printf("The current AD value %d = 0x%08X   \r\n",0,ADC_RegSimult_ConvertedValue[0]); 
  ADC_RegSimult_ConvertedValue_Show[0] = temp0 ;
  ADC_RegSimult_ConvertedValue_Show[1] = temp1  ;

	ADC_RegSimult_ConvertedValueLocal[0] = (float)temp0 /4096*3.3; 
	ADC_RegSimult_ConvertedValueLocal[1] = (float)temp1 /4096*3.3;
 		 
	printf("The current AD value %d = 0x%04X   %f V\r\n",0,ADC_RegSimult_ConvertedValue_Show[0],ADC_RegSimult_ConvertedValueLocal[0]); 
  printf("The current AD value %d = 0x%04X   %f V\r\n",1,ADC_RegSimult_ConvertedValue_Show[1],ADC_RegSimult_ConvertedValueLocal[1]); 
	printf("\r\n\r\n");
  GDScream_Data_show( ADC_RegSimult_ConvertedValue_Show,NOFCHANEL);
	#endif 
     
}

  

3:main.c

/*******************************************************************************
* @file    GPIO/JTAG_Remap/main.c 
* @author  MCD Application Team
* @version V3.5.0
* @date    08-April-2011
* @brief   Main program body
******************************************************************************
* @attention
*    
*    
******************************************************************************
*/ 

/* Includes ------------------------------------------------------------------*/
#include "stm32f10x.h"
#include "PROJ_book.h" 

 

/**
  * @brief  Main program.
  * @param  None
  * @retval None
  */
void delay(int x);
void fn_LED_Flash_Init(void);
void fn_usart_show_Init(void);
void fn_DMA_show_Init(void);
void fn_I2C_EE_Init(void);
void fn_I2C_EE_Soft_Init(void);
void fn_SPI_FLASH_Soft_Init(void);
void fn_FatFs_Document_Init(void);
void fn_SRAM_Init(void);
void fn_LCD_Init(void);
void fn_XPT2046_Init(void);
void fn_Adc_Init(void);

#define countof(a)      (sizeof(a) / sizeof(*(a)))
  
#define  _I2C_BufferSize (countof(writeData)-1)
static uint8_t writeData[_I2C_PageSize]={4,5,6,7,8,9,10,11};
static uint8_t writeData2[_I2C_PageSize]={24,25,26,27,28,29,30,31};
static uint8_t ReadData[_I2C_BufferSize]={0};

#define  _SPI_BufferSize  SPI_PAGE_SIZE   //(countof(write_SPI_Data)-1)
static uint8_t write_SPI_Data[_SPI_BufferSize]={0};
static uint8_t Read_SPI_Data[_SPI_BufferSize]={0};

int main(void)
{  
      int16_t sAD_X, sAD_Y ;
      fn_Adc_Init(); 							//ADC 采集
      //*************LCD系統初始化**************  
      fn_XPT2046_Init(); 
      fn_Lcd_Page_Init();  
      fn_Lcd_Page4();
      while(1){     				 
	    ADC_get_value(); 	 
      }
}

 
//======================================================================
//======================================================================

void fn_LCD_Init(void){             //LCD運行測試
    __IO int16_t int_check;
    fn_Systick_Delay(500,_Systick_ms);
    int_check = (__IO int16_t)ILI9341_Init();
    Display_LCD_clear();
      switch(int_check){
        case 0x05:
            printf("\n-->LCD 運行正常\r\n");
            Lcd_display_String("  LCD 運行正常\r\n");     
            break;
        default:
            printf("\n-->LCD 運行異常\r\n");
            Lcd_display_String("   LCD 運行異常\r\n\r\n");
            
      }       
}

//======================================================================
//======================================================================

void fn_XPT2046_Init(void){             //XPT2046運行測試
   printf("\n-->LCD 觸摸初始化\r\n");
   Lcd_display_String("  LCD 觸摸初始化\r\n");   
   XPT2046_GPIO_Init();  
//   while(XPT2046_Touch_Calibrate_Page() == 0){;}
   Lcd_display_String("  LCD 觸摸初始化ok\r\n");
}
//======================================================================
//======================================================================


void fn_Adc_Init(void){ 
		ADCx_Init();
}
 
/******************* (C) COPYRIGHT 2011 STMicroelectronics *****END OF FILE****/


顯示屏:

4: XPT2046_LCD_GridDiagram_book.h

#ifndef      __XPT2046_LCD_GRIDDIAGRAM_BOOK__
#define	     __XPT2046_LCD_GRIDDIAGRAM_BOOK__

#include "stm32f10x.h"
															 
#include "XPT2046_LCD_Function_book.h"
#include "XPT2046_LCD_book.h"
#include "LCD_book.h"
#include "LCD_Draw_book.h" 
#include "XPT2046_LCD_Device_book.h"
#include "ADC_book.h"
#include "USART_book.h"

 
//======================================================形式是默認還是定制化=====================================================
/**

模式0:				.		模式1:		.	模式2:			.	模式3:					
					A		.					A		.		A					.		A									
					|		.					|		.		|					.		|							
					Y		.					X		.		Y					.		X					
					0		.					1		.		2					.		3					
	<--- X0 o		.	<----Y1	o		.		o 2X--->  .		o 3Y--->	
------------------------------------------------------------	
模式4:				.	模式5:			.	模式6:			.	模式7:					
	<--- X4 o		.	<--- Y5 o		.		o 6X--->  .		o 7Y--->	
					4		.					5		.		6					.		7	
					Y		.					X		.		Y					.		X						
					|		.					|		.		|					.		|							
					V		.					V		.		V					.		V		
---------------------------------------------------------				
											 LCD屏示例
								|---------|-----------------------------|
								|	AD測試	| 													  |
								|					|													    |
								|					|													    |
								|					|													    |
								|					|													    |
								|					|													    |
								|					|													    |
								|					|										    			|
								|					|										    			|
								|---------|-----------------------------|
								屏幕正面(寬240,高320)270 50 -2
**/
 
///******************************* XPT2046 觸摸屏參數定義 ***************************/
 
 
#define	 XPT2046_GDCHANNEL_X 	       ILI9341_MORE_PIXEL 	          //通道Y+的選擇控制字	
#define	 XPT2046_GDCHANNEL_Y 	       ILI9341_LESS_PIXEL	            //通道X+的選擇控制字

#define  LCD_GDWork_X_LENGTH         270
#define  LCD_GDNwork_Y_LENGTH        238

#define  LCD_GDControl_X_Start       0
#define  LCD_GDControl_Y_Start       0
#define  LCD_GDControl_X_LENGTH      (XPT2046_GDCHANNEL_X - LCD_GDWork_X_LENGTH - 2)
#define  LCD_GDControl_Y_LENGTH      XPT2046_GDCHANNEL_Y

#define  LCD_GDSCAN_X_Start          (LCD_GDControl_X_LENGTH+1)
#define  LCD_GDSCAN_Y_Start          1
#define  LCD_GDSCAN_X_LENGTH         LCD_GDWork_X_LENGTH
#define  LCD_GDSCAN_Y_LENGTH         LCD_GDNwork_Y_LENGTH
#define  LCD_GDSCAN_X_End          	 (XPT2046_GDCHANNEL_X-1)
#define  LCD_GDSCAN_Y_End          	 (XPT2046_GDCHANNEL_Y-1)
#define  Center_lineNum 						 6

#define GDBUTTON_NUM     3


typedef struct{
	int16_t value_x_before ; 
	int16_t value_x_now ;
	int16_t value_y_before ; 
	int16_t value_y_now ;
	uint32_t para_color;	
	uint16_t data_value; 
  void (*draw_point)(void *draw_pot);     //按鍵描繪函數
   
}AD_Point;

void GDScream_Data_show(int32_t *data_value , uint16_t data_num);

 

#endif




5: XPT2046_LCD_GridDiagram_book.c

/**
  ******************************************************************************
  * @file    palette.c
  * @author  fire
  * @version V1.0
  * @date     
  * @brief   觸摸畫板應用函數
  ******************************************************************************
  * @attention
  *
  ******************************************************************************
  */

#include "XPT2046_LCD_GridDiagram_book.h"
#include  
#include 

Touch_Button GDbutton[GDBUTTON_NUM];

static void GDTouch_Button_Init(void);
static void Draw_btn_GDcontrol_Button(void *btn); 
static void Btn_command_GDcontrol_Button(void);
static void GDControl_Text_Init(void);
static void GDScream_show_Init(void);
static void GDScream_Line_Init(void);
static void GDScream_Data_show_init(void);
static void Draw_point(void * draw_pot);
 /**
* @brief  Palette_Init 畫板初始化
* @param  無
* @retval 無
*/
#define _LCD_GDSCAN_MODE   LCD_SCAN_MODE_3
void GridDiagram_Init(void){
	uint16_t datax , datay;
  ILI9341_GramScan(_LCD_GDSCAN_MODE);
  
  //初始化畫板顏色
  LCD_SetBackColor(CL_GREY2);
  ILI9341_Clear(0,0,XPT2046_GDCHANNEL_X+1,XPT2046_GDCHANNEL_Y+1);
  
  
  LCD_SetColors(CL_YELLOW , CL_WHITE);

 
  ILI9341_DrawRectangle(1,1,LCD_GDControl_X_LENGTH-1,LCD_GDControl_Y_LENGTH-1,0,1);
	LCD_SetColors(CL_WHITE , CL_WHITE);	
					 
  ILI9341_DrawRectangle(LCD_GDControl_X_Start ,
                        LCD_GDControl_Y_Start ,                    
                        LCD_GDControl_X_LENGTH, 
                        LCD_GDControl_Y_LENGTH,0,1);
   
  LCD_SetColors(CL_BLACK , CL_WHITE);						 
  ILI9341_DrawRectangle(LCD_GDSCAN_X_Start,
                        LCD_GDSCAN_Y_Start,                    
                        LCD_GDSCAN_X_LENGTH, 
                        LCD_GDSCAN_Y_LENGTH,1,1);
  GDScream_Line_Init();
  GDScream_Data_show_init();
  //初始化按鈕
  GDTouch_Button_Init();
  GDControl_Text_Init();
  //對屬性物件進行刷新
  GDScream_show_Init();
  
}


/**
* @brief  GDTouch_Button_Init 畫板初始化
* @param  無
* @retval 無
*/
static void GDTouch_Button_Init(void){
	uint8_t i ;
	for(i = 0 ;i<=GDBUTTON_NUM ;i++ ){
    GDbutton[i].start_x = BUTTON_START_X+3; 
    GDbutton[i].end_x = GDbutton[i].start_x+COLOR_BLOCK_WIDTH ;
    GDbutton[i].start_y = (COLOR_BLOCK_HEIGHT + 20)*(i+1);
    GDbutton[i].end_y = GDbutton[i].start_y + COLOR_BLOCK_HEIGHT ;    
    GDbutton[i].touch_flag = 0;  
    GDbutton[i].draw_btn = Draw_btn_GDcontrol_Button ;
    GDbutton[i].btn_command = Btn_command_GDcontrol_Button ;
  }
  GDbutton[0].para = CL_GREY; //構建按鈕的屬性
	GDbutton[1].para = CL_GREY; //構建按鈕的屬性
	GDbutton[2].para = CL_GREY; //構建按鈕的屬性
	//描繪按鈕
  for(i=0 ;itouch_flag == 0){
    //背景為功能鍵的顏色
    LCD_SetColors(ptr->para , CL_WHITE);
    ILI9341_DrawRectangle(ptr->start_x , ptr->start_y,\
                          ptr->end_x - ptr->start_x,\
												  ptr->end_y - ptr->start_y,1,1);
     
    //白色背景邊框
    LCD_SetColors(CL_BOX_BORDER1 , CL_WHITE);
    ILI9341_DrawRectangle(ptr->start_x , ptr->start_y,\
                          ptr->end_x - ptr->start_x,\
												  ptr->end_y - ptr->start_y,0,2);
  }else{//按鍵按下
    //白色背景
    LCD_SetColors(CL_WHITE , CL_WHITE);
    ILI9341_DrawRectangle(ptr->start_x , ptr->start_y,\
                          ptr->end_x - ptr->start_x,\
												  ptr->end_y - ptr->start_y,1,1);
     //白色背景邊框
    LCD_SetColors(CL_BOX_BORDER2 , CL_WHITE);
    ILI9341_DrawRectangle(ptr->start_x , ptr->start_y,\
                          ptr->end_x - ptr->start_x,\
												  ptr->end_y - ptr->start_y,0,2);
  }
}


/**
* @brief  Btn_command_GDcontrol_Button 畫板初始化
* @param  無
* @retval 無
*/
static void Btn_command_GDcontrol_Button(void){
				 ;
}


/**
* @brief  Draw_btn_GDcontrol_Button 畫板初始化
* @param  無
* @retval 無
*/
static void GDControl_Text_Init(void){
		LCD_SetColors(CL_RED,CL_WHITE);
	/*選擇字體,使用中英文顯示時,盡量把英文選擇成8*16的字體,
	*中文字體大小是16*16的,需要其它字體請自行制作字模*/
	/*這個函數只對英文字體起作用*/
	LCD_SetFont(&Font8x16);
	ILI9341_DispString_EN_CH( LCD_GDControl_X_Start +2, LCD_GDControl_Y_Start + 5, "AD檢查");

  	LCD_SetColors(CL_BLACK,CL_WHITE);
	/*選擇字體,使用中英文顯示時,盡量把英文選擇成8*16的字體,
	*中文字體大小是16*16的,需要其它字體請自行制作字模*/
	/*這個函數只對英文字體起作用*/
	LCD_SetFont(&Font8x16);
	ILI9341_DispString_EN_CH( LCD_GDControl_X_Start +2, LCD_GDControl_Y_Start + 25, " 王琪");
}


/**
* @brief  Draw_btn_GDcontrol_Button 畫板初始化
* @param  無
* @retval 無
*/


static void GDScream_Line_Init(void){
	uint8_t i ;	
	uint16_t center_linexstart , center_lineystart ,center_linexend , center_lineyend;
  uint16_t center_linespace  ;
	center_linexstart = LCD_GDSCAN_X_Start ;
	center_lineystart = XPT2046_GDCHANNEL_Y / 2;
	center_linexend   = LCD_GDSCAN_X_End;
	center_lineyend	= center_lineystart ;
  
  center_linespace = (XPT2046_GDCHANNEL_Y - 2 )/Center_lineNum ;
													 

  LCD_SetColors(CL_RED , CL_WHITE); 
  
  ILI9341_DrawLine( center_linexstart ,center_lineystart ,center_linexend ,center_lineyend,1 );
	LCD_SetColors(CL_YELLOW , CL_WHITE); 
	ILI9341_DrawLine( center_linexstart ,center_lineystart + center_linespace ,center_linexend ,center_lineyend + center_linespace,1 );
	ILI9341_DrawLine( center_linexstart ,center_lineystart - center_linespace ,center_linexend ,center_lineyend - center_linespace,1 );
	LCD_SetColors(CL_BLUE , CL_WHITE);
  ILI9341_DrawLine( center_linexstart ,center_lineystart + (center_linespace*2) ,center_linexend ,center_lineystart + (center_linespace*2),1 );
	ILI9341_DrawLine( center_linexstart ,center_lineystart - (center_linespace*2) ,center_linexend ,center_lineystart - (center_linespace*2),1 );
}

/**
* @brief  Draw_btn_GDcontrol_Button 畫板初始化
* @param  無
* @retval 無
*/
static void GDScream_show_Init(void){
	uint8_t i ;	 
  for( i=0 ; i< (GDBUTTON_NUM); i++ ) {
		GDbutton[i].draw_btn(&GDbutton[i]);
	}
}

//===============================================關于屏幕畫點的程序函數============================================== 

// 屏幕正面(寬320,高240) 

//   270 個像素
//   238 / 6  38   
//   500MS  一個數字  
//   像素間隔 Dx = 2 = 135 個數字
//   可以測試85S 的數據
//   4096 12 AD   每個像素高度為 20 *38 = 760  大約 0.6V  精度為0.02V   
//   設定變量 :uint16 data_value /  DX = 2  DY =  1 HX= 1 HY = 20 , uint16 data_value_before   data_value_now  ,    
	
#define   DX   1 
#define   DY   1 
#define   HX   1 
#define   HY   20
  
__IO int16_t data_value_x_before = -1;    //之前的狀態點
__IO int16_t data_value_x_now = -1;       // 現在的狀態點
__IO int16_t data_value_y_before = -1; 
__IO int16_t data_value_y_now = -1;

AD_Point  ad_point[NOFCHANEL];      //采樣數據模塊

static void GDScream_Data_show_init(void){       //顯示數據AD的樣式初始化
  uint8_t i=0 ;
	data_value_x_before = -1 ;
	data_value_x_now = -1;
	data_value_y_before = -1 ;
	data_value_y_now = -1;
	for( i=0;ivalue_x_before) == -1){
			ILI9341_SetPointPixel(ptr_point->value_x_now,ptr_point->value_y_now,1);
	}else{
		  ILI9341_DrawLine(ptr_point->value_x_before,ptr_point->value_y_before,ptr_point->value_x_now,ptr_point->value_y_now,1);
	}
}

void  GDScream_Data_show(int32_t *data_value , uint16_t data_num){  //GD顯示 
  
  uint8_t i=0 , j=0 ;
  char cStr [ 100 ];	
	char * pStr = 0;
   float ADC_ValueLocal;
 
    int16_t data_value_x , data_value_y ;


	for(i=0 ; i LCD_GDSCAN_X_End-2)){   // 如果是第一次進入最左端存儲點就刷新程序
		LCD_SetColors(CL_BLACK , CL_WHITE);						 
		ILI9341_DrawRectangle(LCD_GDSCAN_X_Start,
                        LCD_GDSCAN_Y_Start,                    
                        LCD_GDSCAN_X_LENGTH, 
                        LCD_GDSCAN_Y_LENGTH,1,1);
		GDScream_Line_Init();

    GDScream_Data_show_init();
 
	} 
  if(data_value_x_before == -1){  //說明是第一次     如果是初始化第一次需要在最左端繪制描繪點
		 for(i=0 ;i;i++){>

審核編輯:湯梓紅

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

    關注

    113

    文章

    6264

    瀏覽量

    185289
  • adc
    adc
    +關注

    關注

    98

    文章

    6510

    瀏覽量

    544987
  • 內存
    +關注

    關注

    8

    文章

    3034

    瀏覽量

    74132
收藏 人收藏

    評論

    相關推薦

    STM32F103×8/STM32F103×B MCU手冊

    1. Q: STM32F103C8T6如何正確配置時鐘系統?A: STM32F103C8T6的時鐘系統配置通常涉及HSE(高速外部時鐘)、HSI(高速內部時鐘)、PLL(相位鎖定環)等。配置時,首先
    發表于 11-18 15:14 ?0次下載

    STM32F407三ADC采樣設置死機怎么解決?

    使用STM32F407三ADCADC1ADC2 ADC3分別分配8個通道(ADC1 把芯片測
    發表于 07-26 06:43

    CKS32F107xx系列MCU的雙重ADC模式

    獨立模式ADC采集需要在一個通道采集并且轉換完成后才會進行下一個通道的采集。而雙重
    的頭像 發表于 07-22 09:19 ?512次閱讀
    CKS32<b class='flag-5'>F</b>107xx系列MCU的<b class='flag-5'>雙重</b><b class='flag-5'>ADC</b><b class='flag-5'>模式</b>

    STM32F103xC,STM32F103xD,STM32F103xE中文資料

    電子發燒友網站提供《STM32F103xC,STM32F103xD,STM32F103xE中文資料.pdf》資料免費下載
    發表于 06-17 14:12 ?4次下載

    STM32F103進入睡眠模式或者待機模式或者停機模式,IO腳原先設置的電平值是否會改變?

    STM32F103進入睡眠模式或者待機模式或者停機模式,IO腳原先設置的電平值是否會改變? 鎖定STM32F103的IO腳,那么當
    發表于 05-17 09:22

    stm32f302和stm32f103ADC區別是什么?

    請問stm32f302和stm32f103ADC除了stm32f103有2個ADCADC1和
    發表于 05-15 08:03

    關于STM32F103使用FSMC同步模式問題求解

    大家好,我現在正在使用STM32F103的FSMC同步模式。 需要讀取高速ADC數據,ADC是8位并口,但需要一個時鐘去觸發轉換,在時鐘上升
    發表于 04-17 08:13

    如何減少STM32F103 ADC采集時間?

    環境: STM32CubeMX 5.6.1(庫為:STM32F11.8.0) IAR for ARM 8.40.2 STM32F103C8T6 BluePill 問題: 使用PA1~PA4進行4
    發表于 04-10 08:20

    stm32f103ADC同步規則模式兩個ADC轉換的通道數量能不一樣嗎?

    stm32f103rct6,配置成雙ADC同步規則模式時,主ADC1、從
    發表于 04-10 06:21

    STM32F103采集模擬量只要低于3.3V,采集的值就是0怎么解決?

    利用STM32F103 ADC直接采集模擬量,現場返修板出現的問題。 采集電壓引腳不論是多少,只要低于3.3V,通過keil仿真直接觀察ADC
    發表于 04-08 07:24

    STM32F4 ADC采集數據不匹配是什么原因造成的?如何解決?

    通過STM32F4 進行同步規則AD采集,設置ADC1,A
    發表于 04-02 08:21

    想用STM32F373的三個SDADC同步采集電壓信號,DMA是工作在什么模式?怎么配置?

    我想用STM32F373的三個SDADC同步采集電壓信號,請問如果要搭配DMA工作,DMA是工作在什么模式?怎么配置?(沒有看到像F4系列的
    發表于 03-28 06:49

    STM32F407的規則同步ADC采集如何實現256k采樣+轉換速度?

    F407的規則同步ADC采集如何 實現256k采樣+轉換速度?
    發表于 03-07 07:51

    ARM系列STM32F103芯片的解密方法

    本文介紹ARM系列STM32F103芯片的解密方法,其內核是Cortex-M3,內存從16K-512K都有。
    發表于 02-28 11:20 ?1737次閱讀

    stm32f103 flash模擬eeprom

    STM32F103是意法半導體(STMicroelectronics)推出的一款32位單片機系列,該系列芯片具有高性能和豐富的外設接口,廣泛應用于工業控制、消費電子、汽車電子等領域。其中
    的頭像 發表于 01-09 11:21 ?2098次閱讀
    主站蜘蛛池模板: 黄色网址中文字幕| 国产精品看片| 成年啪啪网站免费播放看| 99久久伊人一区二区yy5099| 视频在线免费| 国产精品www夜色影视| 97涩涩涩| 中文字幕视频二区| 免费aⅴ网站| 一级特一级特色生活片| 羞涩妩媚玉腿呻吟嗯啊销魂迎合| 天天摸日日舔| 免费看日本黄色片| 色网站免费在线观看| 日韩精品在线一区二区| 免费视频你懂得| 成人亚洲电影| 国产亚洲婷婷香蕉久久精品 | 2022国产情侣真实露脸在线| 天天做天天摸天天爽天天爱| 日本色视| 欧美成人性色区| 精品久久久久久国产免费了| 欧美三四级片| 一及黄色| 怡红院网址| 色聚网久久综合| 久国产精品久久精品国产四虎| 97综合| 夜夜夜爽爽爽久久久| 欧美黄色成人| 一区二区高清在线| 性生大片一级毛片免费观看| 你懂的亚洲| 午夜影院入口| 五月婷婷六月丁香在线| 中文天堂最新版在线精品| 日本黄色一级大片| 99成人在线| 国产精品视频久久久久久| 免费看18污黄|