/* Includes ------------------------------------------------------------------*/
#include "main.h" // inclui definições principais do projeto gerado pelo STM32CubeIDE
#include <math.h>
/* -------------------- Constantes de aplicação -------------------- */
#define DISPLAY_DIGIT_COUNT 4 // quantidade de dígitos do display multiplexado
#define DISPLAY_REFRESH_TICKS 5 // atualiza a multiplexação a cada 5 ms
#define COUNTER_UPDATE_TICKS 1000 // incrementa o contador a cada 1000 ms
#define COUNTER_MAX_VALUE 10000 // contador de 0 a 9999
#define DIGIT_ON_LEVEL GPIO_PIN_RESET // nível lógico que ativa um dígito
#define DIGIT_OFF_LEVEL GPIO_PIN_SET // nível lógico que desativa um dígito
/* Índices da tabela de códigos do display */
#define DIGIT_0 0 // índice do dígito 0 na tabela de códigos
#define DIGIT_1 1 // índice do dígito 1 na tabela de códigos
#define DIGIT_2 2 // índice do dígito 2 na tabela de códigos
#define DIGIT_3 3 // índice do dígito 3 na tabela de códigos
#define DIGIT_4 4 // índice do dígito 4 na tabela de códigos
#define DIGIT_5 5 // índice do dígito 5 na tabela de códigos
#define DIGIT_6 6 // índice do dígito 6 na tabela de códigos
#define DIGIT_7 7 // índice do dígito 7 na tabela de códigos
#define DIGIT_8 8 // índice do dígito 8 na tabela de códigos
#define DIGIT_9 9 // índice do dígito 9 na tabela de códigos
#define DIGIT_OFF 10 // índice que representa todos os segmentos apagados
#define BETA 3950
#define HISTERESIS 2
/* USER CODE BEGIN PV */
ADC_HandleTypeDef hadc1;
/* -------------------- Mapeamento de hardware -------------------- */
/* Segmentos do display em GPIOA: a..g + dp */
static const uint16_t segmentPins[8] = {
GPIO_PIN_0, // a
GPIO_PIN_1, // b
GPIO_PIN_2, // c
GPIO_PIN_3, // d
GPIO_PIN_4, // e
GPIO_PIN_5, // f
GPIO_PIN_6, // g
GPIO_PIN_7 // dp
};
/* Seleção dos 4 dígitos em GPIOB */
static const uint16_t digitSelectPins[DISPLAY_DIGIT_COUNT] = {
GPIO_PIN_12, // unidade
GPIO_PIN_13, // dezena
GPIO_PIN_14, // centena
GPIO_PIN_15 // milhar
};
static const uint8_t seg7Codes[11] = {
0x3F, // 0
0x06, // 1
0x5B, // 2
0x4F, // 3
0x66, // 4
0x6D, // 5
0x7D, // 6
0x07, // 7
0x7F, // 8
0x6F, // 9
0x00 // apagado
};
volatile uint32_t contador = 0; // contador principal; volatile indica que pode mudar em contexto de interrupção
/* digitos[0] = unidade, digitos[1] = dezena, etc. */
volatile uint8_t digitos[DISPLAY_DIGIT_COUNT] = {0, 0, 0, 0}; // vetor com os quatro dígitos atualmente exibidos
/* USER CODE END PV */
/* Private function prototypes -----------------------------------------------*/
/* USER CODE BEGIN PFP */
void SystemClock_Config(void); // protótipo da função de configuração do clock do sistema
static void MX_GPIO_Init(void); // protótipo da função de inicialização dos GPIOs
static void MX_ADC1_Init(void);
static void Display_WriteCode(uint8_t codeIndex); // escreve um padrão de segmentos
static void Display_DisableAllDigits(void); // desabilita todos os dígitos do display
static void Display_EnableDigit(uint8_t digitIndex); // habilita apenas um dígito do display
static void Display_Update(void); // executa um passo da multiplexação
static void Counter_UpdateDigits(void); // converte o contador em unidade/dezena/centena/milhar
static void Counter_Increment(void); // incrementa o contador com retorno para zero
static void Display_InitState(void); // inicializa o display em estado conhecido
/*-------------------------VARIAVEIS NOVAS------------------------------*/
volatile uint16_t temperatura = 0;
volatile uint16_t Tmin = 20; // valor inicial
volatile uint16_t Tmax = 50;
volatile uint8_t modo_config = 0; // começa mostrando a temp
volatile uint8_t modo = 0; // 0 = temp, 1 = tmin, 2 = tmax
// LEDs
#define LED_AQUECIMENTO GPIO_PIN_3
#define LED_RESFRIAMENTO GPIO_PIN_4
/* -------------------- Funções auxiliares -------------------- */
/* Escreve no barramento de segmentos o código correspondente ao índice informado */
static void Display_WriteCode(uint8_t codeIndex)
{
uint8_t segmentCode = seg7Codes[codeIndex]; // obtém o padrão binário correspondente ao dígito desejado
for (uint8_t i = 0; i < 8; i++) // percorre os 8 bits: a, b, c, d, e, f, g e dp
{
if (segmentCode & (1U << i)) // testa se o bit i está em nível 1
{
HAL_GPIO_WritePin(GPIOA, segmentPins[i], GPIO_PIN_SET); // liga o segmento correspondente
}
else
{
HAL_GPIO_WritePin(GPIOA, segmentPins[i], GPIO_PIN_RESET); // desliga o segmento correspondente
}
}
}
/* Desabilita todos os dígitos do display multiplexado */
static void Display_DisableAllDigits(void)
{
HAL_GPIO_WritePin(
GPIOB,
digitSelectPins[0] | digitSelectPins[1] | digitSelectPins[2] | digitSelectPins[3], // forma máscara com os 4 dígitos
DIGIT_OFF_LEVEL // coloca todos no nível lógico de desativação
);
}
/* Habilita apenas um dígito */
static void Display_EnableDigit(uint8_t digitIndex)
{
HAL_GPIO_WritePin(GPIOB, digitSelectPins[digitIndex], DIGIT_ON_LEVEL); // ativa somente o dígito indicado
}
/* Atualiza um único dígito por chamada (multiplexação) */
static void Display_Update(void)
{
static uint8_t currentDigit = 0; // guarda qual dígito será atualizado nesta chamada
/* Evita ghosting: primeiro apaga segmentos e desabilita todos os dígitos */
Display_WriteCode(DIGIT_OFF); // apaga os segmentos antes de trocar de dígito
Display_DisableAllDigits(); // desabilita todos os dígitos para evitar sobreposição visual
/* Habilita o dígito desejado e escreve seu valor */
Display_EnableDigit(currentDigit); // ativa o dígito atual da multiplexação
Display_WriteCode(digitos[currentDigit]); // escreve o valor correspondente naquele dígito
currentDigit++; // avança para o próximo dígito
if (currentDigit >= DISPLAY_DIGIT_COUNT) // ao atingir a quantidade total de dígitos...
{
currentDigit = 0; // ...reinicia a sequência
}
}
/* Atualiza o vetor de dígitos a partir do valor atual do contador */
static void Counter_UpdateDigits(void)
{
digitos[0] = contador % 10; // extrai a unidade
digitos[1] = (contador / 10) % 10; // extrai a dezena
digitos[2] = (contador / 100) % 10; // extrai a centena
digitos[3] = (contador / 1000) % 10; // extrai o milhar
}
/* Incrementa o contador com retorno para zero ao atingir 10000 */
static void Counter_Increment(void)
{
Counter_UpdateDigits(); // atualiza os dígitos exibidos após a mudança do valor
}
/* Coloca o display em estado inicial conhecido */
static void Display_InitState(void)
{
Display_DisableAllDigits(); // garante que nenhum dígito fique ativo no início
Display_WriteCode(DIGIT_OFF); // apaga todos os segmentos
Counter_UpdateDigits(); // sincroniza o vetor de dígitos com o valor inicial do contador
}
/* -------------------- Callbacks -------------------- */
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
static uint32_t last_interrupt_time = 0;
if (HAL_GetTick() - last_interrupt_time < 200) return;
last_interrupt_time = HAL_GetTick();
// BOTÃO 1 → diminui Tmin e Tmax
if (GPIO_Pin == GPIO_PIN_13)
{
if (modo==1)
{
Tmin--;
if(Tmin<0)
Tmin++;
}
else if (modo==2)
{
Tmax--;
if(Tmax==Tmin)
Tmax++;
}
}
// BOTÃO 2 → aumenta Tmin e Tmax
if (GPIO_Pin == GPIO_PIN_14)
{
if (modo==1)
{
Tmin++;
if(Tmin == Tmax)
Tmin--;
}
else if (modo==2)
{
Tmax++;
if(Tmax > 80)
Tmax = 80;
}
}
// BOTÃO 3 → alterna modo
if (GPIO_Pin == GPIO_PIN_15)
{
modo++;
if(modo > 3) modo = 0;
}
}
void HAL_SYSTICK_Callback(void)
{
static uint8_t displayTickCount = 0; // conta ticks até o próximo passo da multiplexação
static uint16_t counterTickCount = 0; // conta ticks até completar aproximadamente 1 segundo
displayTickCount++; // incrementa a contagem usada na atualização do display
if (displayTickCount >= DISPLAY_REFRESH_TICKS) // verifica se chegou o momento de atualizar a multiplexação
{
displayTickCount = 0; // reinicia a contagem da multiplexação
Display_Update(); // atualiza um dos dígitos do display
}
}
/* NEW FUNCITIONS */
static uint16_t Converter_Temperatura(uint16_t adc)
{
float Vref = 3.3;
float Rfixo = 10000.0;
float T0 = 298.15; // 25°C em Kelvin
float R0 = 10000.0;
float Vout = (adc / 4095.0) * Vref;
float Rntc = Rfixo * (Vout / (Vref - Vout));
float tempK = 1.0 / ( (1.0 / T0) + (1.0 / BETA) * log(Rntc / R0) );
float tempC = tempK - 273.15;
return (uint16_t)tempC;
}
static void Controle_Temperatura(void)
{
if (temperatura < Tmin - HISTERESIS)
{
// liga aquecimento
HAL_GPIO_WritePin(GPIOB, LED_AQUECIMENTO, GPIO_PIN_SET);
HAL_GPIO_WritePin(GPIOB, LED_RESFRIAMENTO, GPIO_PIN_RESET);
}
else if (temperatura > Tmax + HISTERESIS)
{
// liga resfriamento
HAL_GPIO_WritePin(GPIOB, LED_AQUECIMENTO, GPIO_PIN_RESET);
HAL_GPIO_WritePin(GPIOB, LED_RESFRIAMENTO, GPIO_PIN_SET);
}
else
{
// dentro da faixa
HAL_GPIO_WritePin(GPIOB, LED_AQUECIMENTO, GPIO_PIN_RESET);
HAL_GPIO_WritePin(GPIOB, LED_RESFRIAMENTO, GPIO_PIN_RESET);
}
}
int main(void)
{
HAL_Init(); // inicializa a HAL, configura a base de tempo e realiza ajustes iniciais do microcontrolador
MX_ADC1_Init();
/* Configure the system clock */
SystemClock_Config(); // configura o clock principal do sistema
/* Initialize all configured peripherals */
MX_GPIO_Init(); // inicializa os pinos GPIO conforme configuração feita no CubeMX
/* USER CODE BEGIN 2 */
Display_InitState(); // coloca o display em estado inicial previsível
/* USER CODE END 2 */
HAL_ADC_Start(&hadc1); // inicia a conversão
HAL_ADC_PollForConversion(&hadc1, HAL_MAX_DELAY); // aguarda terminar a primeira
while (1)
{
uint16_t adc = HAL_ADC_GetValue(&hadc1);
temperatura = Converter_Temperatura(adc);
if (modo > 0)
{
HAL_GPIO_WritePin(GPIOB, LED_AQUECIMENTO, GPIO_PIN_SET);
HAL_GPIO_WritePin(GPIOB, LED_RESFRIAMENTO, GPIO_PIN_SET);
// mostra Tmin ou Tmax alternando
if(modo==1)
contador = Tmin;
if(modo==2)
contador = Tmax;
}
else
{
// mostra temperatura atual
contador = temperatura;
Controle_Temperatura();
}
Counter_UpdateDigits();
HAL_Delay(200);
}
/* USER CODE END 3 */
}
void SystemClock_Config(void)
{
RCC_OscInitTypeDef RCC_OscInitStruct = {0}; // estrutura para configuração do oscilador
RCC_ClkInitTypeDef RCC_ClkInitStruct = {0}; // estrutura para configuração dos clocks do barramento
/** Initializes the RCC Oscillators according to the specified parameters
* in the RCC_OscInit TypeDef structure.
*/
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE; // seleciona o oscilador externo de alta velocidade
RCC_OscInitStruct.HSEState = RCC_HSE_ON; // habilita o HSE
RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1; // define o pré-divisor do HSE como 1
RCC_OscInitStruct.HSIState = RCC_HSI_ON; // mantém também o oscilador interno habilitado
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON; // habilita o PLL
RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE; // usa o HSE como fonte do PLL
RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9; // multiplica a frequência por 9
if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK) // aplica a configuração do oscilador
{
Error_Handler(); // em caso de erro, chama a rotina de tratamento
}
/** Initializes the CPU, AHB and APB buses clocks
*/
RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
|RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2; // define quais clocks serão configurados
RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK; // escolhe o PLL como clock do sistema
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1; // AHB = SYSCLK
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2; // APB1 = HCLK/2
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1; // APB2 = HCLK
if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK) // aplica a configuração dos clocks
{
Error_Handler(); // trata erro de configuração
}
}
static void MX_ADC1_Init(void)
{
/* USER CODE BEGIN ADC1_Init 0 */
/* USER CODE END ADC1_Init 0 */
ADC_ChannelConfTypeDef sConfig = {0};
/* USER CODE BEGIN ADC1_Init 1 */
/* USER CODE END ADC1_Init 1 */
/** Common config
*/
hadc1.Instance = ADC1;
hadc1.Init.ScanConvMode = ADC_SCAN_DISABLE;
hadc1.Init.ContinuousConvMode = ENABLE;
hadc1.Init.DiscontinuousConvMode = DISABLE;
hadc1.Init.ExternalTrigConv = ADC_SOFTWARE_START;
hadc1.Init.DataAlign = ADC_DATAALIGN_RIGHT;
hadc1.Init.NbrOfConversion = 1;
if (HAL_ADC_Init(&hadc1) != HAL_OK)
{
Error_Handler();
}
/** Configure Regular Channel
*/
sConfig.Channel = ADC_CHANNEL_8;
sConfig.Rank = ADC_REGULAR_RANK_1;
sConfig.SamplingTime = ADC_SAMPLETIME_55CYCLES_5;
if (HAL_ADC_ConfigChannel(&hadc1, &sConfig) != HAL_OK)
{
Error_Handler();
}
/* USER CODE BEGIN ADC1_Init 2 */
/* USER CODE END ADC1_Init 2 */
}
static void MX_GPIO_Init(void)
{
GPIO_InitTypeDef GPIO_InitStruct = {0}; // estrutura usada para configurar os pinos GPIO
/* USER CODE BEGIN MX_GPIO_Init_1 */
/* USER CODE END MX_GPIO_Init_1 */
/* GPIO Ports Clock Enable */
__HAL_RCC_GPIOC_CLK_ENABLE(); // habilita o clock da porta C
__HAL_RCC_GPIOD_CLK_ENABLE(); // habilita o clock da porta D
__HAL_RCC_GPIOA_CLK_ENABLE(); // habilita o clock da porta A
__HAL_RCC_GPIOB_CLK_ENABLE(); // habilita o clock da porta B
/*Configure GPIO pin Output Level */
HAL_GPIO_WritePin(GPIOA, L0_Pin|L1_Pin|L2_Pin|L3_Pin
|L4_Pin|L5_Pin|L6_Pin|L7_Pin
|LED_POWER_Pin, GPIO_PIN_RESET); // inicializa em nível baixo os pinos de segmentos e LED
/*Configure GPIO pin Output Level */
HAL_GPIO_WritePin(GPIOB, D4_Pin|D3_Pin|D2_Pin|D1_Pin, GPIO_PIN_RESET); // inicializa em nível baixo os pinos de seleção de dígitos
/*Configure GPIO pin : PC13 */
GPIO_InitStruct.Pin = GPIO_PIN_13|GPIO_PIN_14|GPIO_PIN_15; // seleciona os pinos PC13, PC14 e PC15
GPIO_InitStruct.Mode = GPIO_MODE_IT_RISING; // configura como interrupção na borda de subida
GPIO_InitStruct.Pull = GPIO_PULLUP; // habilita resistor de pull-up interno
HAL_GPIO_Init(GPIOC, &GPIO_InitStruct); // aplica configuração à porta C
/*Configure GPIO pins : L0_Pin L1_Pin L2_Pin L3_Pin
L4_Pin L5_Pin L6_Pin L7_Pin
LED_POWER_Pin */
GPIO_InitStruct.Pin = L0_Pin|L1_Pin|L2_Pin|L3_Pin
|L4_Pin|L5_Pin|L6_Pin|L7_Pin
|LED_POWER_Pin ; // seleciona os pinos de saída da porta A
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; // saída digital push-pull
GPIO_InitStruct.Pull = GPIO_NOPULL; // sem pull-up/pull-down
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; // velocidade baixa de comutação
HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); // aplica configuração à porta A
/*Configure GPIO pins : D4_Pin D3_Pin D2_Pin D1_Pin */
GPIO_InitStruct.Pin = D4_Pin|D3_Pin|D2_Pin|D1_Pin; // seleciona os pinos de saída da porta B
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; // saída digital push-pull
GPIO_InitStruct.Pull = GPIO_NOPULL; // sem pull-up/pull-down
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; // velocidade baixa de comutação
HAL_GPIO_Init(GPIOB, &GPIO_InitStruct); // aplica configuração à porta B
/* EXTI interrupt init*/
HAL_NVIC_SetPriority(EXTI15_10_IRQn, 0, 0); // define prioridade da interrupção EXTI das linhas 10 a 15
HAL_NVIC_EnableIRQ(EXTI15_10_IRQn); // habilita a interrupção EXTI15_10 no NVIC
/* USER CODE BEGIN MX_GPIO_Init_2 */
GPIO_InitStruct.Pin = GPIO_PIN_3 | GPIO_PIN_4;
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_InitStruct.Pull = GPIO_NOPULL;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
HAL_GPIO_WritePin(GPIOB, GPIO_PIN_3 | GPIO_PIN_4, GPIO_PIN_RESET);
/* USER CODE END MX_GPIO_Init_2 */
}
/* USER CODE BEGIN 4 */
/* USER CODE END 4 */
/**
* @brief This function is executed in case of error occurrence.
* @retval None
*/
void Error_Handler(void)
{
/* USER CODE BEGIN Error_Handler_Debug */
/* User can add his own implementation to report the HAL error return state */
__disable_irq(); // desabilita todas as interrupções para parar o sistema em condição de erro
while (1)
{
// permanece em loop infinito para facilitar depuração
}
/* USER CODE END Error_Handler_Debug */
}
#ifdef USE_FULL_ASSERT
/**
* @brief Reports the name of the source file and the source line number
* where the assert_param error has occurred.
* @param file: pointer to the source file name
* @param line: assert_param error line source number
* @retval None
*/
void assert_failed(uint8_t *file, uint32_t line)
{
/* USER CODE BEGIN 6 */
/* User can add his own implementation to report the file name and line number,
ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
/* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */PA8