Предварительный процессор: # если сравнивать два значения - PullRequest
0 голосов
/ 06 сентября 2018

Я работал над настройкой UART для микроконтроллера ATmega168. Мой первый код не работал. Во время проверки я наткнулся на файл заголовка, используемый в работающей программе. Он использует директивы препроцессора.

Это было довольно странно для меня, поэтому я решил запустить его в компиляторе c и проверить вывод.

#include <stdio.h>

#define F_CPU 1000000UL

#ifndef BAUD                          /* if not defined in Makefile... */
#define BAUD  9600                     /* set a safe default baud rate */
#endif

#define BAUD_TOL 2
#define UBRR_VALUE (((F_CPU) + 8UL * (BAUD)) / (16UL * (BAUD)) -1UL)

#if 100 * (F_CPU) > (16 * ((UBRR_VALUE) + 1)) * (100 * (BAUD) + (BAUD) * (BAUD_TOL))
#define USE_2X 1
#define TEST 1

#elif 100 * (F_CPU) < (16 * ((UBRR_VALUE) + 1)) * (100 * (BAUD) - (BAUD) * (BAUD_TOL))
#define USE_2X 1
#define TEST 11

#else
#define USE_2X 0
#define TEST 111

#endif


int main(void) {

    printf("UBRR_VALUE: %ld\n", UBRR_VALUE);
    printf("USE_2X: %d\n", USE_2X);
    printf("%ld\n", (16 * ((UBRR_VALUE) + 1)) * (100 * (BAUD) - (BAUD) * (BAUD_TOL)));
    printf("F_CPU * 100 = %ld\n", (100*F_CPU));
    printf("TEST = %d\n", TEST);
    return 0;
}

Выход:

UBRR_VALUE: 6
USE_2X: 1
105369600
F_CPU * 100 = 100000000
TEST = 11

, поскольку 100 * F_CPU меньше (16 * ((UBRR_VALUE) + 1)) * (100 * (BAUD) - (BAUD) * (BAUD_TOL)), TEST присваивается значение 11.

Но это я меняю F_CPU на 16000000UL:

UBRR_VALUE: 103
USE_2X: 0
1565491200
F_CPU * 100 = 1600000000
TEST = 111

Здесь также 100 * F_CPU меньше (16 * ((UBRR_VALUE) + 1)) * (100 * (BAUD) - (BAUD) * (BAUD_TOL)).

Но почему TEST равно 111?

Редактировать: * Вот 100 * F_CPU больше (16 * ((UBRR_VALUE) + 1)) * (100 * (BAUD) - (BAUD) * (BAUD_TOL))

1 Ответ

0 голосов
/ 06 сентября 2018

Макросы расширяются, как описано ниже:

Для второго случая F_CPU до 16000000UL (большее значение)

100 * (F_CPU) > (16 * ((UBRR_VALUE) + 1)) * (100 * (BAUD) + (BAUD) * (BAUD_TOL)) 

оценивается как 0x5f5e1000 > 0x611e8000, что ложно

100 * (F_CPU) < (16 * ((UBRR_VALUE) + 1)) * (100 * (BAUD) - (BAUD) * (BAUD_TOL))

оценивается как 0x5f5e1000 < 0x5d4f8000, что также неверно.

Следовательно, применимы следующие макросы:

#define USE_2X 0
#define TEST 111
...