对串口接收FIFO处理机制的解读

李肖遥 2022-09-13 22:08
    关注、星标公众号,直达精彩内容

素材来源:https://blog.csdn.net/weixin_44386927/article/details/124626454

整理:技术让梦想更伟大 | 李肖遥



一、FIFOFIFO

First Input First Output的缩写,先入先出队列。

使用的场景:一般是在不同的时钟域之间的数据传输(简单理解即:一个(读\写)快,另外的一个(读\写)慢的场景中。)

本质操作:就是将收的数据存储的一个线性的数组中,通过指针指向该数组的自加1(偏移)来遍历数据,达到读取或者写入的目的。

作用:起到缓冲环节,可防止数据的丢失。

对数据量大的进行存储,避免频繁的总线操作。并且可满足dma的操作。

在fifo中需要理解连个重要成员:

宽度:指一次写读操作的数据位数

深度:存储多少个宽度的数据。(如:存储8个16位宽的数据)。

第一类、FIFO处理机制如下:

FIFO信息的定义:

/*
该结构体定义成员有
缓存区,
长度,
输出,
输入的计数。
*/

typedef struct fifo_t {
    uint8_t *buf;
 uint32_t size;
 uint32_t in;
 uint32_t out;
} _fifo_str;
#define min(x,y) ((x) < (y)?(x):(y))  
1234567891011121314

1、初始化FIFO

fifo_str fifo_str;

int FifoInit(uint8_t *fifo_addr, uint32_t fifo_size)//初始化fifo
{
 _fifo_str *p = &fifo_str;
 
 if(fifo_addr == NULL || fifo_size == 0)//判断数据是否为空
  return -1;

 memset((char *)p, 0sizeof(_fifo_str));//初始化结构体
 p->buf = fifo_addr;//对应宽度
    p->in = 0;//输入计数
    p->out = 0;//输出计数
    p->size = fifo_size;//对应深度
 return 0;
}
12345678910111213141516

2、数据的长度获取

//数据的实际使用数据空间长度
int FifoDataLen(void)
{
 _fifo_str *p = &fifo_str;
 return (p->in - p->out);//输入计数-输出计数
}
//剩余数据空间长度
int FifoSpaceLen(void)
{
 _fifo_str *p = &fifo_str;
 
 return (p->size - (p->in - p->out));//定义长度-(实际长度)
}
12345678910111213

3、FIFO的进和出处理

//获取fifo数据
//数据的内容缓存区,要读的长度
int FifoRead(uint8_t *buf, uint32_t len)
{
 uint32_t i = 0, j = 0;
 _fifo_str *p = &fifo_str;

 j = (p->out % p->size);//获取剩余空间长度未读量
 len = min(len, p->in - p->out);//防止长度为超出实际拥有的数据长度,即让读取的长度在  (0<设定len<定义的缓存区长度len )这间的实际长度
 i = min(len, p->size - j);//获取实际内容的长度,的数据长度
 memcpy(buf, p->buf + j, i);//将数据通道里的数据拷贝给缓存区
 memcpy(buf + i, p->buf, len - i);//将未有数据的区域存入,对应为写入数据的区域数据(即,有数据的填数据,没数据的地方补0)
 p->out += len;//已读的数据量
 return len;//实际读到的数据长度
}
//对fifo写入数据
int FifoWrite(uint8_t *buf, uint32_t len)
{
 uint32_t i = 0, j = 0;
 _fifo_str *p = &fifo_str;

 j = p->in % p->size;//获取要写入的剩余空间长度
 len = min(len, p->size - p->in + p->out);//得到实际写入的长度
 i = min(len, p->size - j);//实际写入数据的长度
 memcpy(p->buf + j, buf, i);//将写入的数据的内容拷贝值数据中。
 memcpy(p->buf, buf + i, len - i);//补充多余空间的内容
 p->in += len;//记录实际写入数据的数量
 return len;//返回写入的长度
}

123456789101112131415161718192021222324252627282930

4、置位记录量

//清空fifo 中的记录量
void FifoClear(void)
{
 _fifo_str *p = &fifo_str;
    p->in = 0;
    p->out = 0;
}
1234567

5、应用处理机制

#define LEN 2048
uint8_t pdata[LEN] = {0};
FifoInit(pdata, LEN);//初始化FIFO
uint8_t buf[32] = {0}; 
int tx_len = 0;
uint8_t tx_buf[100] = {0};

HAL_UART_Receive_IT(&huart1, buf, IT_LEN);//串口的接收中断开启
while(1)
 { 
  tx_len = FifoDataLen();//获取数据长度
  if (tx_len > 0)
  {
   tx_len = (tx_len > 100) ? 100 : tx_len;//判读数据长度是否越界
   FifoRead(tx_buf, tx_len);//读取在中断中写入FIFO缓存的数据
   HAL_UART_Transmit(&huart1, tx_buf, tx_len, 1000);//将读到的数据通过串口发送出来
  }
 }

接收回调函数中的处理
HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
 if (FifoSpaceLen() >= 串口记录的接收数据长度)//判断写入FIFO空间的数据量是否大于接收的数据量
 {
  FifoWrite(huart->pRxBuffPtr, huart->RxXferCount);//想FIFO中写入数据
 }

123456789101112131415161718192021222324252627

该FIFO的处理机制中用的记录是通过uint32t类型进行记录的,可能在遇到超出其数据极限的情况,导致数据通信异常。(该类型的数据极限较大,为特殊情况可能出现的情况)。

第二类、FIFO处理机制如下:

/* 定义串口波特率和FIFO缓冲区大小,
分为发送缓冲区和接收缓冲区*/
 
#if UART1_FIFO_EN == 1  
#define UART1_BAUD 115200  
#define UART1_TX_BUF_SIZE 1*1024  
#define UART1_RX_BUF_SIZE 1*1024
#endif 

/* 串口设备结构体
 设置发送、接收缓存区(长度),
 并设置两个变量,一个是指针,一个是计数
 */
 
typedef struct 
{
  
 USART_TypeDef *uart; /* STM32内部串口设备指针 */  

 uint8_t *pTxBuf; /* 发送缓冲区 */ 
  uint8_t *pRxBuf; /* 接收缓冲区 */ 
  uint16_t usTxBufSize; /* 发送缓冲区大小 */  
 uint16_t usRxBufSize; /* 接收缓冲区大小 */

   __IO uint16_t usTxWrite; /* 发送缓冲区写指针 */ 
  __IO uint16_t usTxRead; /* 发送缓冲区读指针 */  
  __IO uint16_t usTxCount; /* 等待发送的数据个数 */ 
 
   __IO uint16_t usRxWrite; /* 接收缓冲区写指针 */ 
   __IO uint16_t usRxRead; /* 接收缓冲区读指针 */
   __IO uint16_t usRxCount; /* 还未读取的新数据个数 */ 
  
 void (*SendBefor)(void); /* 开始发送之前的回调函数指针(主要用于RS485切换到发送模式) */  
 void (*SendOver)(void); /* 发送完毕的回调函数指针(主要用于RS485将发送模式切换为接收模式) */ 
 void (*ReciveNew)(uint8_t _byte); /* 串口收到数据的回调函数指针 */  
 uint8_t Sending; /* 正在发送中 */

 }UART_T;

/* 定义每个串口结构体变量 */ 
#if UART1_FIFO_EN == 1
static UART_T g_tUart1; 
static uint8_t g_TxBuf1[UART1_TX_BUF_SIZE]; /* 发送缓冲区 */  
static uint8_t g_RxBuf1[UART1_RX_BUF_SIZE]; /* 接收缓冲区 */ 
#endif

12345678910111213141516171819202122232425262728293031323334353637383940414243

怎样才叫做回调函数?回调函数,是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数, 当这个指针被用为调用它所指向的函数时, 我们就说这是回调函数。

区别指针函数和函数指针:

//指针函数:
int *fun(int x,int y)
int *x=fun(4,5);
在调用指针函数时,需要同类型的指针来接收函数返回值
是函数,返回值时指针
属于数据类型
123456
//函数指针:
int (*fun)(int x,int y)
int x(int x,int y);
x=fun;
fun=&x;
x=(*fun)(13);
是指针,指向函数。
属于函数名称
12345678

1.初始化串口FIFO对应的相关的变量

static void UartVarInit(void) 
{  
 #if UART1_FIFO_EN == 1 
  g_tUart1.uart = USART1; /* STM32 串口设备 */ 
  g_tUart1.pTxBuf = g_TxBuf1; /* 发送缓冲区指针 */ 
  g_tUart1.pRxBuf = g_RxBuf1; /* 接收缓冲区指针 */ 
  g_tUart1.usTxBufSize = UART1_TX_BUF_SIZE; /* 发送缓冲区大小 */ 
  g_tUart1.usRxBufSize = UART1_RX_BUF_SIZE; /* 接收缓冲区大小 */ 
  g_tUart1.usTxWrite = 0/* 发送FIFO写索引 */  
 g_tUart1.usTxRead = 0/* 发送FIFO读索引 */  
 g_tUart1.usRxWrite = 0/* 接收FIFO写索引 */  
 g_tUart1.usRxRead = 0/* 接收FIFO读索引 */  
 g_tUart1.usRxCount = 0/* 接收到的新数据个数 */  
 g_tUart1.usTxCount = 0/* 待发送的数据个数 */  
 g_tUart1.SendBefor = 0/* 发送数据前的回调函数 */  
 g_tUart1.SendOver = 0/* 发送完毕后的回调函数 */  
 g_tUart1.ReciveNew = 0/* 接收到新数据后的回调函数 */  
 g_tUart1.Sending = 0/* 正在发送中标志 */ 
 #endif 

1234567891011121314151617181920

明确中断服务程序的顺序:中断函数处理:void USART1_IRQHandler(void)—》 UART中断请求:HAL_UART_IRQHandler(UART_HandleTypeDef *huart)—》 中断使能:UART_Receive_IT— 》 中断回调函数 HAL_UART_RxCpltCallback(huart);

#if UART1_FIFO_EN == 1
void USART1_IRQHandler(void) //系统中串口的中断函数入口
{
    UartIRQ(&g_tUart1);
}
#endif

1234567

2.编辑自定义的UART中断请求

static void UartIRQ(UART_T *_pUart)
{
    uint32_t isrflags = READ_REG(_pUart->uart->ISR);
    uint32_t cr1its = READ_REG(_pUart->uart->CR1);
    uint32_t cr3its = READ_REG(_pUart->uart->CR3);

    if ((isrflags & USART_ISR_RXNE) != RESET)
    {
        /* 从串口接收数据寄存器读取数据存放到接收FIFO */
        uint8_t ch;
        ch = READ_REG(pUart->uart->RDR);
        /* 读串口接收数据寄存器 */
        _pUart->pRxBuf[_pUart->usRxWrite] = ch;         /* 填入串口接收FIFO */
        if (++_pUart->usRxWrite >= _pUart->usRxBufSize) /* 接收FIFO的写指针+1 */
        {
            _pUart->usRxWrite = 0;
        }
        if (_pUart->usRxCount < _pUart->usRxBufSize) /* 统计未处理的字节个数 */
        {
            _pUart->usRxCount++;
        }
        /* 回调函数,通知应用程序收到新数据,一般是发送1个消息或者设置一个标记 */
        // if (_pUart->usRxWrite == _pUart->usRxRead)
        // if (_pUart->usRxCount == 1)
        {
            if (_pUart->ReciveNew)
            {
                _pUart->ReciveNew(ch); /* 比如,交给MODBUS解码程序处理字节流 */
            }
        }
    }
    /* 处理发送缓冲区空中断 */
    if (((isrflags & USART_ISR_TXE) != RESET) && (cr1its & USART_CR1_TXEIE) != RESET)
    {
        // if (_pUart->usTxRead == _pUart->usTxWrite)
        if (_pUart->usTxCount == 0/* 发送缓冲区已无数据可取 */
        {
            /* 发送缓冲区的数据已取完时, 禁止发送缓冲区空中断 (注意:此时最后1个数据还未真正发送完毕)*/
            // USART_ITConfig(_pUart->uart, USART_IT_TXE, DISABLE);
            CLEAR_BIT(_pUart->uart->CR1, USART_CR1_TXEIE); /* 使能数据发送完毕中断 */
            // USART_ITConfig(_pUart->uart, USART_IT_TC, ENABLE);
            SET_BIT(_pUart->uart->CR1, USART_CR1_TCIE);
        }
        Else /* 还有数据等待发送 */
        {
            _pUart->Sending = 1/* 从发送FIFO取1个字节写入串口发送数据寄存器 */
            // USART_SendData(_pUart->uart, _pUart->pTxBuf[_pUart->usTxRead]);
            _pUart->uart->TDR = _pUart->pTxBuf[_pUart->usTxRead];
            if (++_pUart->usTxRead >= _pUart->usTxBufSize)
            {
                _pUart->usTxRead = 0;
            }
            _pUart->usTxCount--;
        }
    }
    /* 数据bit位全部发送完毕的中断 */
    if (((isrflags & USART_ISR_TC) != RESET) && ((cr1its & USART_CR1_TCIE) != RESET))
    {
        // if (_pUart->usTxRead == _pUart->usTxWrite)
        if (_pUart->usTxCount == 0)
        { /* 如果发送FIFO的数据全部发送完毕,禁止数据发送完毕中断 */
            // USART_ITConfig(_pUart->uart, USART_IT_TC, DISABLE);
            CLEAR_BIT(_pUart->uart->CR1, USART_CR1_TCIE); /* 回调函数, 一般用来处理RS485通信,将RS485芯片设置为接收模式,避免抢占总线 */
            if (_pUart->SendOver)
            {
                _pUart->SendOver();
            }
            _pUart->Sending = 0;
        }

        else
        { /* 正常情况下,不会进入此分支 */
            /* 如果发送FIFO的数据还未完毕,则从发送FIFO取1个数据写入发送数据寄存器 */
            // USART_SendData(_pUart->uart, _pUart->pTxBuf[_pUart->usTxRead]);
            _pUart->uart->TDR = _pUart->pTxBuf[_pUart->usTxRead];
            if (++_pUart->usTxRead >= _pUart->usTxBufSize)
            {
                _pUart->usTxRead = 0;
            }
            _pUart->usTxCount--;
        }
    } /* 清除中断标志 */
    SET_BIT(_pUart->uart->ICR, UART_CLEAR_PEF);
    SET_BIT(_pUart->uart->ICR, UART_CLEAR_FEF);
    SET_BIT(_pUart->uart->ICR, UART_CLEAR_NEF);
    SET_BIT(_pUart->uart->ICR, UART_CLEAR_OREF);
    SET_BIT(_pUart->uart->ICR, UART_CLEAR_IDLEF);
    SET_BIT(_pUart->uart->ICR, UART_CLEAR_TCF);
    SET_BIT(_pUart->uart->ICR, UART_CLEAR_LBDF);
    SET_BIT(_pUart->uart->ICR, UART_CLEAR_CTSF);
    SET_BIT(_pUart->uart->ICR, UART_CLEAR_CMF);
    SET_BIT(_pUart->uart->ICR, UART_CLEAR_WUF);
    SET_BIT(_pUart->uart->ICR, UART_CLEAR_TXFECF);
}
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394

3.填写数据到UART发送缓冲区。

并启动发送中断,中断处理函数发送完毕后,自动关闭发送中断 .

static void UartSend(UART_T *_pUart, uint8_t *_ucaBuf, uint16_t _usLen) 
{  
 uint16_t i;  
  for (i = 0; i < _usLen; i++)  
 {  /* 如果发送缓冲区已经满了,则等待缓冲区空 */ 
   while (1)  
  {  
   __IO uint16_t usCount;   DISABLE_INT(); 
    usCount = _pUart->usTxCount; 
    ENABLE_INT();   
   if (usCount < _pUart->usTxBufSize)  
   {  
    break;  
   }  
   else if(usCount == _pUart->usTxBufSize)/* 数据已填满缓冲区 */  
   { 
     if((_pUart->uart->CR1 & USART_CR1_TXEIE) == 0
     {  
     SET_BIT(_pUart->uart->CR1, USART_CR1_TXEIE);  
    } 
    }  
  }   /* 将新数据填入发送缓冲区 */ 
   _pUart->pTxBuf[_pUart->usTxWrite] = _ucaBuf[i];  
   DISABLE_INT();  
  if (++_pUart->usTxWrite >= _pUart->usTxBufSize) 
   {  
   _pUart->usTxWrite = 0;  
  } 
   _pUart->usTxCount++;  
  ENABLE_INT(); 
  }  
  SET_BIT(_pUart->uart->CR1, USART_CR1_TXEIE); /* 使能发送中断(缓冲区空) */ 


12345678910111213141516171819202122232425262728293031323334

4.向串口发送一组数据。

数据放到发送缓冲区后立即返回,由中断服务程序在后台完成发送

void comSendBuf(COM_PORT_E _ucPort, uint8_t *_ucaBuf, uint16_t _usLen) 
{  
 UART_T *pUart;  
  pUart = ComToUart(_ucPort); 
  if (pUart == 0
  {
    return
  }  
  if (pUart->     != 0
  {
    pUart->SendBefor(); /* 如果是RS485通信,可以在这个函数中将RS485设置为发送模式 */  
 }  
  UartSend(pUart, _ucaBuf, _usLen); 
}

123456789101112131415

向串口发送1个字节。数据放到发送缓冲区后立即返回, 由中断服务程序在后台完成发送

void comSendChar(COM_PORT_E _ucPort, uint8_t _ucByte)
 
{  
 comSendBuf(_ucPort, &_ucByte, 1); 


123456

函数comSendChar是发送一个字节, 通过调用函数comSendBuf实现, 而函数comSendBuf又是通过调用函数UartSend实现, 这个函数是重点。

5.将COM端口号转换为UART指针

UART_T *ComToUart(COM_PORT_E _ucPort) 
{  
 if (_ucPort == COM1)  
 { 
  #if UART1_FIFO_EN == 1  
  return &g_tUart1; 
   #else 
   return 0;  
  #endif 
  }  
 else  
 {  
  Error_Handler(__FILE__, __LINE__);  
  return 0
  } 
}

1234567891011121314151617

6.从串口接收缓冲区读取1字节数据

static uint8_t UartGetChar(UART_T *_pUart, uint8_t *_pByte) 
{  

 uint16_t usCount;   /* usRxWrite 变量在中断函数中被改写,主程序读取该变量时,必须进行临界区保护 */  

 DISABLE_INT();  usCount = _pUart->usRxCount;  ENABLE_INT();   /* 如果读和写索引相同,则返回0 */ 
 
 //if (_pUart->usRxRead == usRxWrite)  
 if (usCount == 0/* 已经没有数据 */  
 {  
  return 0
  } 
  else  
 {  
  *_pByte = _pUart->pRxBuf[_pUart->usRxRead]; /* 从串口接收  FIFO取1个数据 */   
  /* 改写FIFO读索引 */  

  DISABLE_INT();
 
   if (++_pUart->usRxRead >= _pUart->usRxBufSize) 
   {  
   _pUart->usRxRead = 0;  
  }  
  _pUart->usRxCount--; 
   ENABLE_INT();  
  return 1;  
 }
 

1234567891011121314151617181920212223242526272829

从接收缓冲区读取1字节,非阻塞。无论有无数据均立即返回。

uint8_t comGetChar(COM_PORT_E _ucPort, uint8_t *_pByte) 
{

 UART_T *pUart;   
 pUart = ComToUart(_ucPort); 
  if (pUart == 0)
 {  
  return 0;  
 } 
   return UartGetChar(pUart, _pByte); 
}
123456789101112

接收数据的调用顺序是:comGetChar–》UartGetChar

7.判断发送缓冲区是否为空

uint8_t UartTxEmpty(COM_PORT_E _ucPort)
{
   UART_T *pUart;
   uint8_t Sending;
   
   pUart = ComToUart(_ucPort);
   if (pUart == 0)
   {
      return 0;
   }

   Sending = pUart->Sending;

   if (Sending != 0)
   {
      return 0;
   }
   return 1;
}

1234567891011121314151617181920

8.清零串口接收缓冲区

void comClearRxFifo(COM_PORT_E _ucPort)
{
 UART_T *pUart;
 pUart = ComToUart(_ucPort);
 if (pUart == 0)
 {
  return;
 }
 pUart->usRxWrite = 0;
 pUart->usRxRead = 0;
 pUart->usRxCount = 0;
}
123456789101112

9.清零串口发送缓冲区

void comClearTxFifo(COM_PORT_E _ucPort)
{
 UART_T *pUart;

 pUart = ComToUart(_ucPort);
 if (pUart == 0)
 {
  return;
 }

 pUart->usTxWrite = 0;
 pUart->usTxRead = 0;
 pUart->usTxCount = 0;
}
1234567891011121314

10.输入输出的重定向

int fputc(int ch, FILE *f)
{
#if 1 /* 将需要printf的字符通过串口中断FIFO发送出去,printf函数会立即返回 */
 comSendChar(COM1, ch);
 return ch;
#else /* 采用阻塞方式发送每个字符,等待数据发送完毕 */
 /* 写一个字节到USART1 */
 USART1->DR = ch;
 /* 等待发送结束 */
 while((USART1->SR & USART_SR_TC) == 0)
 {}
 return ch;
#endif
}


int fgetc(FILE *f)
{
#if 1 /* 从串口接收FIFO中取1个数据, 只有取到数据才返回 */
 uint8_t ucData;
 while(comGetChar(COM1, &ucData) == 0);
 return ucData;
#else
 /* 等待接收到数据 */
 while((USART1->SR & USART_SR_RXNE) == 0)
 {}
 return (int)USART1->DR;
#endif
}
1234567891011121314151617181920212223242526272829

11.应用层初始化:

//FIFO串口初始化
UartVarInit(void);
//串口参数配置
void bsp_SetUartParam(USART_TypeDef *Instance,  uint32_t BaudRate, uint32_t Parity, uint32_t Mode)
{
 UART_HandleTypeDef UartHandle; 
 /*##-1- 配置串口硬件参数 ######################################*/
 /* 异步串口模式 (UART Mode) */
 /* 配置如下:
   - 字长    = 8 位
   - 停止位  = 1 个停止位
   - 校验    = 参数Parity
   - 波特率  = 参数BaudRate
   - 硬件流控制关闭 (RTS and CTS signals) */

 UartHandle.Instance        = Instance;
 UartHandle.Init.BaudRate   = BaudRate;
 UartHandle.Init.WordLength = UART_WORDLENGTH_8B;
 UartHandle.Init.StopBits   = UART_STOPBITS_1;
 UartHandle.Init.Parity     = Parity;
 UartHandle.Init.HwFlowCtl  = UART_HWCONTROL_NONE;
 UartHandle.Init.Mode       = Mode;
 UartHandle.Init.OverSampling = UART_OVERSAMPLING_16;
    
 if (HAL_UART_Init(&UartHandle) != HAL_OK)
 {
  Error_Handler(__FILE__, __LINE__);
 }
}
//对应的串口波特率配置
void comSetBaud(COM_PORT_E _ucPort, uint32_t _BaudRate)
{
 USART_TypeDef* USARTx;
 USARTx = ComToUSARTx(_ucPort);
 if (USARTx == 0)
 {
  return;
 }
 bsp_SetUartParam(USARTx,  _BaudRate, UART_PARITY_NONE, UART_MODE_TX_RX);
}
//硬件初始化
void InitHardUart(void)
{
GPIO 复用....
 /* 配置NVIC the NVIC for UART */   
 HAL_NVIC_SetPriority(USART1_IRQn, 01);
 HAL_NVIC_EnableIRQ(USART1_IRQn);
  
 /* 配置波特率、奇偶校验 */
 bsp_SetUartParam(USART1,  UART1_BAUD, UART_PARITY_NONE, UART_MODE_TX_RX);

 CLEAR_BIT(USART1->SR, USART_SR_TC);   /* 清除TC发送完成标志 */
    CLEAR_BIT(USART1->SR, USART_SR_RXNE); /* 清除RXNE接收标志 */
 // USART_CR1_PEIE | USART_CR1_RXNEIE
 SET_BIT(USART1->CR1, USART_CR1_RXNEIE); /* 使能PE. RX接受中断 */
}
在主循环中

comGetChar(COM1, &read);//获取一个字节数据
comSendBuf(COM1, (uint8_t *)buf, strlen(buf));//发送数据
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061


版权声明:本文来源网络,免费传达知识,版权归原作者所有。如涉及作品版权问题,请联系我进行删除。

‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧  END  ‧‧‧‧‧‧‧‧‧‧‧‧‧‧‧

关注我的微信公众号,回复“加群”按规则加入技术交流群。


点击“阅读原文”查看更多分享,欢迎点分享、收藏、点赞、在看。

李肖遥 公众号“技术让梦想更伟大”,作者:李肖遥,专注嵌入式,只推荐适合你的博文,干货,技术心得,与君共勉。
评论
  • ARMv8-A是ARM公司为满足新需求而重新设计的一个架构,是近20年来ARM架构变动最大的一次。以下是对ARMv8-A的详细介绍: 1. 背景介绍    ARM公司最初并未涉足PC市场,其产品主要针对功耗敏感的移动设备。     随着技术的发展和市场需求的变化,ARM开始扩展到企业设备、服务器等领域,这要求其架构能够支持更大的内存和更复杂的计算任务。 2. 架构特点    ARMv8-A引入了Execution State(执行状
    丙丁先生 2025-01-12 10:30 112浏览
  • LVGL(Light and Versatile Graphics Library)是一个免费的开源图形库,旨在为各种微控制器(MCU)和微处理器(MPU)创建美观的用户界面(UI)。LVGL可以在占用很少资源的前提下,实现丝滑的动画效果和平滑滚动的高级图形,具有轻量化、跨平台可用性、易于移植、操作友好以及免费使用等诸多优势。近期,飞凌嵌入式为OK3506J-S开发板移植了最新9.2版本的LVGL,支持多种屏幕构件以及鼠标、键盘、触摸等多种输入方式, 能够带来更加友好的操作界面;同时,启动速度也
    飞凌嵌入式 2025-01-10 10:57 28浏览
  • 随着数字化的不断推进,LED显示屏行业对4K、8K等超高清画质的需求日益提升。与此同时,Mini及Micro LED技术的日益成熟,推动了间距小于1.2 Pitch的Mini、Micro LED显示屏的快速发展。这类显示屏不仅画质卓越,而且尺寸适中,通常在110至1000英寸之间,非常适合应用于电影院、监控中心、大型会议、以及电影拍摄等多种室内场景。鉴于室内LED显示屏与用户距离较近,因此对于噪音控制、体积小型化、冗余备份能力及电气安全性的要求尤为严格。为满足这一市场需求,开关电源技术推出了专为
    晶台光耦 2025-01-13 10:42 114浏览
  • 在不断发展的电子元件领域,继电器——作为切换电路的关键设备,正在经历前所未有的技术变革。固态继电器(SSR)和机械继电器之间的争论由来已久。然而,从未来发展的角度来看,固态继电器正逐渐占据上风。本文将从耐用性、速度和能效三个方面,全面剖析固态继电器为何更具优势,并探讨其在行业中的应用与发展趋势。1. 耐用性:经久耐用的设计机械继电器:机械继电器依靠物理触点完成电路切换。然而,随着时间的推移,这些触点因电弧、氧化和材料老化而逐渐磨损,导致其使用寿命有限。因此,它们更适合低频或对切换耐久性要求不高的
    腾恩科技-彭工 2025-01-10 16:15 77浏览
  • 新年伊始,又到了对去年做总结,对今年做展望的时刻 不知道你在2024年初立的Flag都实现了吗? 2025年对自己又有什么新的期待呢? 2024年注定是不平凡的一年, 一年里我测评了50余块开发板, 写出了很多科普文章, 从一个小小的工作室成长为科工公司。 展望2025年, 中国香河英茂科工, 会继续深耕于,具身机器人、飞行器、物联网等方面的研发, 我觉得,要向未来学习未来, 未来是什么? 是掌握在孩子们生活中的发现,和精历, 把最好的技术带给孩子,
    丙丁先生 2025-01-11 11:35 108浏览
  • 流量传感器是实现对燃气、废气、生活用水、污水、冷却液、石油等各种流体流量精准计量的关键手段。但随着工业自动化、数字化、智能化与低碳化进程的不断加速,采用传统机械式检测方式的流量传感器已不能满足当代流体计量行业对于测量精度、测量范围、使用寿命与维护成本等方面的精细需求。流量传感器的应用场景(部分)超声波流量传感器,是一种利用超声波技术测量流体流量的新型传感器,其主要通过发射超声波信号并接收反射回来的信号,根据超声波在流体中传播的时间、幅度或相位变化等参数,间接计算流体的流量,具有非侵入式测量、高精
    华普微HOPERF 2025-01-13 14:18 97浏览
  • 根据Global Info Research(环洋市场咨询)项目团队最新调研,预计2030年全球无人机电池和电源产值达到2834百万美元,2024-2030年期间年复合增长率CAGR为10.1%。 无人机电池是为无人机提供动力并使其飞行的关键。无人机使用的电池类型因无人机的大小和型号而异。一些常见的无人机电池类型包括锂聚合物(LiPo)电池、锂离子电池和镍氢(NiMH)电池。锂聚合物电池是最常用的无人机电池类型,因为其能量密度高、设计轻巧。这些电池以输出功率大、飞行时间长而著称。不过,它们需要
    GIRtina 2025-01-13 10:49 84浏览
  • 随着全球向绿色能源转型的加速,对高效、可靠和环保元件的需求从未如此强烈。在这种背景下,国产固态继电器(SSR)在实现太阳能逆变器、风力涡轮机和储能系统等关键技术方面发挥着关键作用。本文探讨了绿色能源系统背景下中国固态继电器行业的前景,并强调了2025年的前景。 1.对绿色能源解决方案日益增长的需求绿色能源系统依靠先进的电源管理技术来最大限度地提高效率并最大限度地减少损失。固态继电器以其耐用性、快速开关速度和抗机械磨损而闻名,正日益成为传统机电继电器的首选。可再生能源(尤其是太阳能和风能
    克里雅半导体科技 2025-01-10 16:18 89浏览
  • PNT、GNSS、GPS均是卫星定位和导航相关领域中的常见缩写词,他们经常会被用到,且在很多情况下会被等同使用或替换使用。我们会把定位导航功能测试叫做PNT性能测试,也会叫做GNSS性能测试。我们会把定位导航终端叫做GNSS模块,也会叫做GPS模块。但是实际上他们之间是有一些重要的区别。伴随着技术发展与越发深入,我们有必要对这三个词汇做以清晰的区分。一、什么是GPS?GPS是Global Positioning System(全球定位系统)的缩写,它是美国建立的全球卫星定位导航系统,是GNSS概
    德思特测试测量 2025-01-13 15:42 70浏览
  • 随着通信技术的迅速发展,现代通信设备需要更高效、可靠且紧凑的解决方案来应对日益复杂的系统。中国自主研发和制造的国产接口芯片,正逐渐成为通信设备(从5G基站到工业通信模块)中的重要基石。这些芯片凭借卓越性能、成本效益及灵活性,满足了现代通信基础设施的多样化需求。 1. 接口芯片在通信设备中的关键作用接口芯片作为数据交互的桥梁,是通信设备中不可或缺的核心组件。它们在设备内的各种子系统之间实现无缝数据传输,支持高速数据交换、协议转换和信号调节等功能。无论是5G基站中的数据处理,还是物联网网关
    克里雅半导体科技 2025-01-10 16:20 101浏览
  • 说到福特,就要从亨利·福特(Henry Ford)这个人物说起。在发明大王爱迪生的电气工厂担任工程师的福特下班后,总是在自家仓库里努力研究和开发汽车。1896年,福特终于成功制造出一辆三轮车,开启了福特汽车的传奇。最初几年,福特都是独自制造汽车并同时进行销售。 (今天很多人都知道的精益管理中的5S方法,或多或少地受到了福特 CANDO方法的影响。)1903年,福特从牧师、律师、银行家、会计师等十一位股东那里筹集了十万美元,并在自家庭院成立了美国第五百零三家汽车公司——福特汽车公司(Fo
    优思学院 2025-01-10 11:21 29浏览
  • 01. 什么是过程能力分析?过程能力研究利用生产过程中初始一批产品的数据,预测制造过程是否能够稳定地生产符合规格的产品。可以把它想象成一种预测。通过历史数据的分析,推断未来是否可以依赖该工艺持续生产高质量产品。客户可能会要求将过程能力研究作为生产件批准程序 (PPAP) 的一部分。这是为了确保制造过程能够持续稳定地生产合格的产品。02. 基本概念在定义制造过程时,目标是确保生产的零件符合上下规格限 (USL 和 LSL)。过程能力衡量制造过程能多大程度上稳定地生产符合规格的产品。核心概念很简单:
    优思学院 2025-01-12 15:43 148浏览
  • 随着国家对环保要求日趋严格。以铅酸电池为动力的电动自行车、电动摩托车,将逐渐受到环保管制。而能量密度更高的磷酸铁锂等锂电池成为优先的选择,锂电池以其高能量密度、快速充电、轻量化等特点,已经大量应用于电动车领域。  光耦在锂电池系统PMU中的应用,能提供完善的安全保护和系统支撑。BMS和电池被封装成安装所需要的尺寸外形,高速的CAN以及RS-485等通信总线,被应用在与控制器、中控之间通信。晶台光耦,被广泛应用于通信隔离、双MCU系统应用地隔离、电机驱动隔离等。下图例举在电动摩
    晶台光耦 2025-01-10 10:44 85浏览
  • 电动汽车(EV)正在改变交通运输,为传统内燃机提供更清洁、更高效的替代方案。这种转变的核心是电力电子和能源管理方面的创新,而光耦合器在其中发挥着关键作用。这些不起眼的组件可实现可靠的通信、增强安全性并优化电动汽车系统的性能,使其成为正在进行的革命中不可或缺的一部分。光耦合器,也称为光隔离器,是一种使用光传输电信号的设备。通过隔离高压和低压电路,光耦合器可确保安全性、减少干扰并保持信号完整性。这些特性对于电动汽车至关重要,因为精确控制和安全性至关重要。 光耦合器在电动汽车中的作用1.电池
    腾恩科技-彭工 2025-01-10 16:14 59浏览
  • Snyk 是一家为开发人员提供安全平台的公司,致力于协助他们构建安全的应用程序,并为安全团队提供应对数字世界挑战的工具。以下为 Snyk 如何通过 CircleCI 实现其“交付”使命的案例分析。一、Snyk 的挑战随着客户对安全工具需求的不断增长,Snyk 的开发团队面临多重挑战:加速交付的需求:Snyk 的核心目标是为开发者提供更快、更可靠的安全解决方案,但他们的现有 CI/CD 工具(TravisCI)运行缓慢,无法满足快速开发和部署的要求。扩展能力不足:随着团队规模和代码库的不断扩大,S
    艾体宝IT 2025-01-10 15:52 155浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦