elbear_arduino_bsp/variants/elsomik/variant.c

194 lines
5.0 KiB
C

/**
*******************************************************************************
* Copyright (c) 2021 Nanjing Qinheng Microelectronics Co., Ltd.
* All rights reserved.
*
* This software component is licensed by WCH under BSD 3-Clause license,
* the "License"; You may not use this file except in compliance with the
* License. You may obtain a copy of the License at:
* opensource.org/licenses/BSD-3-Clause
*
*******************************************************************************
*/
#include "pins_arduino.h"
#include "mik32_hal_adc.h"
#include "wiring_analog.h"
#include "wiring_LL.h"
/**
* @brief Determines the address of the port by the board pin number to which this pin belongs on the MCU
* @return The address of the port corresponding to the board pin number. Can return 0 if the pin not exists
*/
GPIO_TypeDef *digitalPinToPort(uint32_t digitalPinNumber)
{
if (digitalPinNumber < 16)
return GPIO_0;
else if ((digitalPinNumber >= 16) && (digitalPinNumber < 32))
return GPIO_1;
else if ((digitalPinNumber >= 32) && (digitalPinNumber < 40))
return GPIO_2;
else
return NULL;
}
// the function returns a reference to the OUTPUT address of the GPIO register
volatile uint32_t *portOutputRegister(GPIO_TypeDef *GPIO_x)
{
return &GPIO_x->OUTPUT_;
}
// the function returns a reference to the STATE address of the GPIO register
volatile uint32_t *portInputRegister(GPIO_TypeDef *GPIO_x)
{
return &GPIO_x->STATE;
}
// ---------------------- ADC ---------------------- //
// determines the ADC channel number by the board pin number
uint32_t analogInputToChannelNumber(uint32_t PinNumber)
{
uint32_t adcChannel = 0;
switch (PinNumber)
{
case PIN_A0:
adcChannel = ADC_CHANNEL0;
break;
case PIN_A1:
adcChannel = ADC_CHANNEL1;
break;
case PIN_A2:
adcChannel = ADC_CHANNEL2;
break;
case PIN_A3:
adcChannel = ADC_CHANNEL3;
break;
case PIN_A4:
adcChannel = ADC_CHANNEL4;
break;
case PIN_A5:
adcChannel = ADC_CHANNEL5;
break;
case PIN_A6:
adcChannel = ADC_CHANNEL6;
break;
case PIN_A7:
adcChannel = ADC_CHANNEL7;
break;
default:
adcChannel = NC;
}
return adcChannel;
}
// ---------------------- PWM ---------------------- //
// use only if digitalPinHasPWM() == true
#define PWM_PIN_TO_PORT_NUMBER(pin) (((pin) & 16) ? 1 : 0)
// use only if digitalPinHasPWM() == true
// return true if digitalPin configured as pwm
bool digitalPinPwmIsOn(uint8_t digitalPin)
{
uint8_t config = 0;
uint8_t pinShift = digitalPin & 0x3;
if (PWM_PIN_TO_PORT_NUMBER(digitalPin) == 0)
config = PIN_GET_PAD_CONFIG(PORT_0_CFG, pinShift);
else
config = PIN_GET_PAD_CONFIG(PORT_1_CFG, pinShift);
if (config == 2)
return true;
else
return false;
}
bool digitalPinHasPWM(uint8_t digitalPin)
{
return (digitalPin < 32) && ((digitalPin & 0xF) < 4);
}
// function is used only if digitalPinHasPWM() is true
TIMER32_TypeDef *pwmPinToTimer(uint32_t digPinNumber)
{
if (digPinNumber < 16)
return TIMER32_1;
else if ((digPinNumber >= 16) && (digPinNumber < 32))
return TIMER32_2;
else
return NULL;
}
// function is used only if digitalPinHasPWM() is true
HAL_TIMER32_CHANNEL_IndexTypeDef pwmPinToTimerChannel(uint32_t digPinNumber)
{
switch (digPinNumber & 0x3)
{
case 0:
return TIMER32_CHANNEL_0;
case 1:
return TIMER32_CHANNEL_1;
case 2:
return TIMER32_CHANNEL_2;
case 3:
return TIMER32_CHANNEL_3;
default:
return 255;
}
}
// ---------------------- interrupts ---------------------- //
// interrupt table is stored in ram to improve performance
// index = interrupt number. In each row {digitalPinNumber, IntLineValue, IntMuxValue}
uint8_t interruptInfo[EXTERNAL_INTERRUPTS_QTY][3] =
{
{P0_4, GPIO_LINE_4, GPIO_MUX_LINE_4_PORT0_4}, // INT0
{P0_9, GPIO_LINE_5, GPIO_MUX_LINE_5_PORT0_9}, // INT1
{P0_10, GPIO_LINE_2, GPIO_MUX_LINE_2_PORT0_10}, // INT2
{P0_15, GPIO_LINE_7, GPIO_MUX_LINE_7_PORT0_15}, // INT3
{P1_6, GPIO_LINE_6, GPIO_MUX_LINE_6_PORT1_6}, // INT4
{P1_9, GPIO_LINE_1, GPIO_MUX_LINE_1_PORT1_9}, // INT5
{P1_12, GPIO_LINE_0, GPIO_MUX_LINE_0_PORT1_12}, // INT6
{P1_15, GPIO_LINE_3, GPIO_MUX_LINE_3_PORT1_15}, // INT7
};
int8_t digitalPinToGpioIntMux(uint8_t digPinNumber)
{
for (uint8_t i = 0; i < EXTERNAL_INTERRUPTS_QTY; i++)
{
if (interruptInfo[i][0] == digPinNumber)
return interruptInfo[i][2];
}
return NOT_AN_INTERRUPT;
}
int8_t digitalPinToGpioIntLine(uint8_t digPinNumber)
{
for (uint8_t i = 0; i < EXTERNAL_INTERRUPTS_QTY; i++)
{
if (interruptInfo[i][0] == digPinNumber)
return interruptInfo[i][1];
}
return NOT_AN_INTERRUPT;
}
int8_t gpioIntLineToInterrupt(uint32_t gpioIntLine)
{
for (uint8_t i = 0; i < EXTERNAL_INTERRUPTS_QTY; i++)
{
if (interruptInfo[i][1] == gpioIntLine)
return i;
}
return NOT_AN_INTERRUPT;
}
int8_t digitalPinToInterrupt(uint32_t digPinNumber)
{
for (uint8_t i = 0; i < EXTERNAL_INTERRUPTS_QTY; i++)
{
if (interruptInfo[i][0] == digPinNumber)
return i;
}
return NOT_AN_INTERRUPT;
}