初级会员

- 积分
- 76
- 金钱
- 76
- 注册时间
- 2012-4-11
- 在线时间
- 0 小时
|
5金钱
我之前由于没有搞过多通道的AD,所以这次尝试了一下DMA数据传输,刚开始我是把内存地址固定给禁止的,如下图红色的字体,我的AD只能读取一路的数据,后面把内存地址固定给使能了,AD正常了,但是OLED显示总是卡在我的查找索引程序这边,好奇怪。。。我的OLED用的是四线控制方式
static void ADC1_Mode_Config(void)
{
DMA_InitTypeDef DMA_InitStructure;
ADC_InitTypeDef ADC_InitStructure;
/* DMA channel1 configuration */
DMA_DeInit(DMA1_Channel1);
DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address; //ADC地址
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue;//内存地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_BufferSize = 10;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外设地址固定
// DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable; //内存地址固定
DMA_InitStructure.DMA_MemoryInc =DMA_MemoryInc_Enable;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //半字
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //循环传输
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
/* Enable DMA channel1 */
DMA_Cmd(DMA1_Channel1, ENABLE);
/* ADC1 configuration */
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //独立ADC模式
ADC_InitStructure.ADC_ScanConvMode = ENABLE ; //禁止扫描模式,扫描模式用于多通道采集
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; //开启连续转换模式,即不停地进行ADC转换
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //不使用外部触发转换
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //采集数据右对齐
ADC_InitStructure.ADC_NbrOfChannel = 2; //要转换的通道数目1
ADC_Init(ADC1, &ADC_InitStructure);
/*配置ADC时钟,为PCLK2的8分频,即9Hz*/
RCC_ADCCLKConfig(RCC_PCLK2_Div8);
/*配置ADC1的通道11为55. 5个采样周期,序列为1 */
ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_9, 2, ADC_SampleTime_55Cycles5);
/* Enable ADC1 DMA */
ADC_DMACmd(ADC1, ENABLE);
/* Enable ADC1 */
ADC_Cmd(ADC1, ENABLE);
/*复位校准寄存器 */
ADC_ResetCalibration(ADC1);
/*等待校准寄存器复位完成 */
while(ADC_GetResetCalibrationStatus(ADC1));
/* ADC校准 */
ADC_StartCalibration(ADC1);
/* 等待校准完成*/
while(ADC_GetCalibrationStatus(ADC1));
/* 由于没有采用外部触发,所以使用软件触发ADC转换 */
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}
主程序:
int main(void)
{
Init();
SysTick_Init();
USART2_Config();
ADC1_Init();
OLED_Init();
OLED_ShowChinese(0,16,4,"现在温度"); //显示现在的温度
OLED_ShowChinese(113,16,1,"℃"); //显示温度单位
OLED_ShowChinese(0,32,4,"电池电压"); //显示当前电池电压
OLED_ShowChar(120,32,'V',16,1); //显示电压的单位
OLED_ShowChinese(0,48,4,"现在距离"); //显示现在距离
OLED_ShowString(100,48,"cm"); //显示距离的单位
OLED_Refresh_Gram();
ADC1_Init();
while(DS18B20_Init());//初始化DS18B20,兼检测18B20
while (1)
{
ADC_Temp=ADC_ConvertedValue[0];
  ower_Detect_Value =((float)ADC_Temp/4096*3.3*2)*100;
Distance_Temp=ADC_ConvertedValue[1];
Distance_Value=2547.8/(float)((Distance_Temp*0.805)-10.41)-0.42;
Usart_Send_Data(); //发送当前空气的温湿度值
Temperature_DS18B20=DS18B20_Get_Temp();
Oled_Display_Temperature();
OLED_ShowNum(64,32,Power_Detect_Value/100,2,16); //显示电压的第一位
OLED_ShowChar(80,32,'.',16,1);//显示小数点
OLED_ShowNum(82,32,Power_Detect_Value%100/10,2,16); //显示电压的第二位
OLED_ShowNum(98,32,Power_Detect_Value%100%10,2,16); //显示电压的第三位
OLED_Refresh_Gram(); //更新数据
GPIO_SetBits(GPIOB,GPIO_Pin_9);
Delay_us(1000000);
GPIO_ResetBits(GPIOB,GPIO_Pin_9);
Delay_us(1000000);
}
}
//oled显示汉字程序
//Display_Num:显示汉字的字数
void OLED_ShowChinese(u8 x,u8 y,u8 Display_Num, u8 *p)
{
u8 i;
u8 Index;
#define MAX_CHAR_POSX 122
#define MAX_CHAR_POSY 58
if(x>MAX_CHAR_POSX){x=0;y+=16;}
if(y>MAX_CHAR_POSY){y=x=0;OLED_Clear();}
for (i=0;i<Display_Num;i++)
{
Index=Find_word_position(p);
OLED_Chinese_character(x,y,Index++,32,1);
x+=16;
p+=2;
}
}
uint8_t Find_word_position(uint8_t *P)
{
u8 i;
if( (*P=='x')&&(*(P+1)=='x') ) {return(0xff);}
for(i=0;i<50;i++)
{
if( (*P==Word_index[2*i])&&(*(P+1)==Word_index[2*i+1]) ) 程序死在这里
{
return(i);
}
}
return(1);
}
|
最佳答案
查看完整内容[请看2#楼]
回复【2楼】pergon:
---------------------------------
谢谢你的回答,问题解决了,就是这个BUF值设置大了导致的,但是不知道为什么会出现程序死在那里的情况。
|