2024-12-25 03:49 AM - edited 2024-12-25 04:30 AM
Hi.
I tried using DMA with I2C before using interrupts and I seem to get the same problem that still remains unsolved.
This time around I decided I will try and not use any interrupts to make things a bit simple. However I am still getting the same result and I just can not seem to figure out why my I2C coms get stuck in the middle of a transmission. The Error flag is not going up either.
I am trying to read from the 0x0C (raw_angle) address from a AS5600 IC.
I do this by sending '0x0C' to '0x36'(AS5600 address). On the BusPirate on SNIFFER mode I see the following on the bus line.
[0x6C+
This did not change from the last example. When I take the BusPirate and the AS5600 IC alone and run the command Step1: [0x6C 0x0C]
Step2: [0x6D rr]
I do manage to get some legit bytes referring to what I believe is the raw_angle values.
When I put it all back onto the MCU it just gets stuck WITH OR WITHOUT the BusPirate on the bus.
In the code below the breakpoint will get stuck in the while loop on line 86 " while (!(DMA1->ISR & DMA_ISR_TCIF2)){} // Wait for DMA transfer complete "
I just do not know what I am doing wrong here.
#include <stdint.h>
#include "stm32f030x8.h"
#if !defined(__SOFT_FP__) && defined(__ARM_FP)
#warning "FPU is not initialized, but the project is compiling for an FPU. Please initialize the FPU before use."
#endif
void Clock_Setup(void) {
RCC->CR |= RCC_CR_HSION; // Enable High-Speed Internal Clock
while (!(RCC->CR & RCC_CR_HSIRDY)); // Wait for HSI to stabilize
RCC->CFGR &= ~RCC_CFGR_SW; // Set HSI as SYSCLK
RCC->CFGR |= RCC_CFGR_SW_HSI;
while ((RCC->CFGR & RCC_CFGR_SWS) != RCC_CFGR_SWS_HSI);
RCC->CFGR &= ~RCC_CFGR_HPRE; // AHB Prescaler: Divide by 1
RCC->CFGR &= ~RCC_CFGR_PPRE; // APB Prescaler: Divide by 1
}
void Config_Pin13(){
// Set PC13 as output
GPIOC->MODER &= ~(3U << (13 * 2)); // Clear MODER13[1:0]
GPIOC->MODER |= (1U << (13 * 2)); // Set MODER13[1:0] to 01 (General-purpose output mode)
}
void I2C_DMA_Init(void) {
// Enable clocks for I2C1, DMA, GPIOB
RCC->AHBENR |= RCC_AHBENR_DMA1EN; // Enable DMA clock
RCC->APB1ENR |= RCC_APB1ENR_I2C1EN; // Enable I2C1 clock
RCC->AHBENR |= RCC_AHBENR_GPIOBEN; // Enable GPIOB clock
// Configure GPIOB pins for I2C1 (PB6: SCL, PB7: SDA)
GPIOB->MODER &= ~(GPIO_MODER_MODER6 | GPIO_MODER_MODER7); // Clear mode
GPIOB->MODER |= (GPIO_MODER_MODER6_1 | GPIO_MODER_MODER7_1); // Set to Alternate Function
GPIOB->AFR[0] |= (1 << GPIO_AFRL_AFRL6_Pos) | (1 << GPIO_AFRL_AFRL7_Pos); // AF1 for I2C1
GPIOB->OTYPER |= GPIO_OTYPER_OT_6 | GPIO_OTYPER_OT_7; // Open-drain
GPIOB->OSPEEDR |= GPIO_OSPEEDER_OSPEEDR6 | GPIO_OSPEEDER_OSPEEDR7; // High speed
GPIOB->PUPDR |= GPIO_PUPDR_PUPDR6_0 | GPIO_PUPDR_PUPDR7_0; // Pull-up
// Configure I2C1
I2C1->TIMINGR = 0x20000913; // Configure timing for 8 MHz HSI and 100 kHz I2C
I2C1->CR1 = I2C_CR1_PE; // Enable I2C1
// Configure DMA for I2C1
// TX DMA (I2C1_TX -> Memory to Peripheral)
DMA1_Channel2->CCR = DMA_CCR_MINC | DMA_CCR_DIR | DMA_CCR_TCIE; // Memory increment, Read from memory
DMA1_Channel2->CPAR = (uint32_t)&I2C1->TXDR; // Peripheral address
// RX DMA (I2C1_RX -> Peripheral to Memory)
DMA1_Channel3->CCR = DMA_CCR_MINC | DMA_CCR_TCIE; // Memory increment
DMA1_Channel3->CPAR = (uint32_t)&I2C1->RXDR; // Peripheral address
I2C1->CR1 |= (I2C_CR1_RXDMAEN | I2C_CR1_TXDMAEN);
}
void I2C_Read_DMA(uint8_t deviceAddr, uint8_t *data, uint16_t size) {
// Configure DMA for RX
DMA1_Channel3->CNDTR = size; // Number of bytes to transfer
DMA1_Channel3->CMAR = (uint32_t)data; // Memory address
DMA1_Channel3->CCR |= DMA_CCR_EN; // Enable DMA channel
// Start I2C transfer
I2C1->CR2 = (deviceAddr << 1) | (size << 16) | I2C_CR2_RD_WRN | I2C_CR2_START; // Set address, size, read mode, and start condition
// Wait for transfer to complete
while (!(DMA1->ISR & DMA_ISR_TCIF3)); // Wait for DMA transfer complete
DMA1->IFCR = DMA_IFCR_CTCIF3; // Clear transfer complete flag
// Wait for STOP condition
while (!(I2C1->ISR & I2C_ISR_STOPF));
I2C1->ICR = I2C_ICR_STOPCF; // Clear STOP flag
// Disable DMA channel
DMA1_Channel3->CCR &= ~DMA_CCR_EN;
}
void I2C_Write_DMA(uint8_t deviceAddr, uint8_t *data, uint16_t size) {
// Configure DMA for TX
DMA1_Channel2->CNDTR = size; // Number of bytes to transfer
DMA1_Channel2->CMAR = (uint32_t)data; // Memory address
DMA1_Channel2->CCR |= DMA_CCR_EN; // Enable DMA channel
// Start I2C transfer
I2C1->CR2 = (deviceAddr << 1) | (size << 16) | I2C_CR2_START; // Set address, size, and start condition
// Wait for transfer to complete
while (!(DMA1->ISR & DMA_ISR_TCIF2)){} // Wait for DMA transfer complete
DMA1->IFCR = DMA_IFCR_CTCIF2; // Clear transfer complete flag
// Wait for STOP condition
while (!(I2C1->ISR & I2C_ISR_STOPF));
I2C1->ICR = I2C_ICR_STOPCF; // Clear STOP flag
// Disable DMA channel
DMA1_Channel2->CCR &= ~DMA_CCR_EN;
}
int main(void) {
uint32_t counter_1 = 0;
uint32_t counter_2 = 0;
uint8_t txData[] = {0x0C};
uint8_t rxData[2];
Clock_Setup(); // Configure the clock
Config_Pin13();
I2C_DMA_Init();
I2C_Write_DMA(0x36, txData, 1);
I2C_Read_DMA(0x36, rxData, 2);
while (1) {
counter_1++;
if (counter_1 == 10000) {
counter_1 = 0;
counter_2++;
if (counter_2==10){
counter_2=0;
GPIOC->ODR ^= (1U << 13);
I2C_Write_DMA(0x36, txData, 1);
}
}
}
}
2024-12-29 11:28 AM
I figured it out.
In my case it seems like the STOP Flag will literally never go high and instead one should rely on the TC Flag. When the STOP gets sent onto the bus the Flag only then goes up. Which is kind of obvious but surprisingly not obvious when tutorials etc show differently.
Thanks