5

私はSTM32 F103xでADCプログラミングを勉強しており、最も単純なケースである単一変換から始めています。内部温度センサー (ADC1 に接続) の値が測定され、USART を使用して COM ポートに送信されます。ターゲットは明確に見えますが、ソース コードをフラッシュにダウンロードしようとすると、COM ポートにデータが送信されません。USART 機能はうまく機能します。完全な変換を待機するループでハングしているため、ADC 構成部分に問題があると思います。

while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET); //Wail for conversion complete

これまでの私のソースコードは次のとおりです。

/* Includes ------------------------------------------------------------*/
#include "stm32f10x.h"
#include <stdio.h>

    uint16_t AD_value;
    const uint16_t V25 = 1750; //when V25=1.41V
    const uint16_t Avg_Slope = 5; //when avg_slc
    uint16_t TemperatureC;

//Define output device
PUTCHAR_PROTOTYPE
{
    USART_SendData(USART1, (uint8_t) ch);
    while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET)
    {}
    return ch;
}

void Usart1Init(void)
{
    GPIO_InitTypeDef GPIO_InitStructure;
    USART_InitTypeDef USART_InitStructure;

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1 | RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE);

    /* COnfig PA9 for USART Tx as alternate function push-pull */
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    /* COnfig PA10 for USART Rx as input floating */
    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
    GPIO_Init(GPIOA, &GPIO_InitStructure);

    /* USARTx configured as follow:
        - BaudRate = 9600 baud  
        - Word Length = 8 Bits
        - One Stop Bit
        - No parity
        - Hardware flow control disabled (RTS and CTS signals)
        - Receive and transmit enabled
    */  
    USART_InitStructure.USART_BaudRate = 9600;
    USART_InitStructure.USART_WordLength = USART_WordLength_8b;
    USART_InitStructure.USART_StopBits = USART_StopBits_1;
    USART_InitStructure.USART_Parity = USART_Parity_No;
    USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
    USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;

    /* USART configuration */
    USART_Init(USART1, &USART_InitStructure);
    USART_Cmd(USART1, ENABLE);  

}

int main(void)
{

    ADC_InitTypeDef ADC_InitStructure;
    Usart1Init();

    RCC_ADCCLKConfig(RCC_PCLK2_Div6); //ADCCLK = PCLK22/6 = 72/6=12MHz
    RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); //Enable ADC1 Clock

    /* ADC1 configuration */
    ADC_DeInit(ADC1); //Power-on default
    ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //Independent conversion mode (single)
    ADC_InitStructure.ADC_ScanConvMode = DISABLE; //Convert single channel only
    ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; //Convert 1 time
    ADC_InitStructure.ADC_ExternalTrigConv = DISABLE; //No external triggering
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //Right 12-bit data alignment
    ADC_InitStructure.ADC_NbrOfChannel = 1; //single channel conversion
    ADC_Init(ADC1, &ADC_InitStructure);
    ADC_TempSensorVrefintCmd(ENABLE); //wake up temperature sensor
    ADC_Cmd(ADC1, ENABLE); //Enable ADC1
    ADC_ResetCalibration(ADC1); //Enable ADC1 reset calibration register
    while(ADC_GetResetCalibrationStatus(ADC1)); //check the end of ADC1 reset calibration register  
    ADC_StartCalibration(ADC1); //Start ADC1 calibration
    while(ADC_GetCalibrationStatus(ADC1)); //Check the end of ADC1 calibration
    ADC_RegularChannelConfig(ADC1, ADC_Channel_16, 1, ADC_SampleTime_1Cycles5); //Select 1.5 cycles conversion for channel 16
    ADC_SoftwareStartConvCmd(ADC1, ENABLE); //Start ADC1 software conversion
    while(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET); //Wail for conversion complete
    AD_value = ADC_GetConversionValue(ADC1); //Read ADC value
    ADC_ClearFlag(ADC1, ADC_FLAG_EOC); //Clear EOC flag

    printf("\r\n ADC value: %d \r\n", AD_value);
    TemperatureC = (uint16_t)((V25-AD_value)/Avg_Slope+25);
    printf("Temperature: %d%cC\r\n", TemperatureC, 176);
    while (1)
    {}  
}

どんなアイデアでも大歓迎です!

4

3 に答える 3

6

ソースコードで何が起こっているのかを説明しようとします。

1-無効の定義はゼロです: typedef enum {DISABLE = 0, ENABLE = !DISABLE} FunctionalState;

2-「ADC_InitStructure.ADC_ExternalTrigConv = DISABLE;」と書くと 実際には、次のようにゼロに割り当てます "ADC_InitStructure.ADC_ExternalTrigConv = 0;

3- これは、ADC_CR2 レジスタの EXTSEL をゼロに割り当てることを意味します。

4- EXTSEL がゼロの場合、ADC 変換は開始するタイマーに依存します。

5- ソフトウェアで adc を開始する場合は、EXTSEL を 0x000E0000 にする必要があります。

6- ご想像のとおり、「ADC_ExternalTrigConv_None」の値は 0x000E0000 として定義されています。

そうwhile(ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET); コードは開始するタイマーに依存するため、ここでハングアップします。開始しないと終了できません。

于 2016-08-08T14:11:53.543 に答える
4

問題が解決しました!外部トリガーを無効にしたのは私のせいです。使用する代わりに:

ADC_InitStructure.ADC_ExternalTrigConv = DISABLE;

次のようになります。

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;

ばかげている!

于 2015-02-04T11:18:18.860 に答える
0

while()ここで、あなたがこれらの 3 つのループのいずれかにぶら下がっていると言って車輪を発明するつもりはありません。それぞれの前にいくつかの出力を作成すると、どこにぶら下がっているかがわかります。

デバッガーをお持ちの場合 (すべての ST 開発ボードに存在し、プログラムのダウンロードに使用している可能性があります)、ためらわずに使用してください :)

于 2015-02-04T08:15:11.707 に答える