源:stm32 DMA数据搬运 [操作寄存器+库函数]

       DMA(Direct Memory Access)常译为“存储器直接存取”。早在Intel的8086平台上就有了DMA应用了。
 
        一个完整的微控制器通常由CPU、存储器和外设等组件构成。这些组件一般在结构和功能上都是独立的,而各个组件的协调和交互就由CPU完成。如此一来,CPU作为整个芯片的核心,其处理的工作量是很大的。如果CPU先从A外设拿到一个数据送给B外设使用,同时C外设又需要D外设提供一个数据。。。这样的数据搬运工作将使CPU的负荷显得相当繁重。
 
       严格的说,搬运数据只是CPU的比较不重要的一种工作。CPU最重要的工作室进行数据运算,从加减乘除到一些高级的运算,包括浮点、积分、微分、FFT等。CPU还需要负责复杂的中断申请和响应,以保证芯片的实时性能。
 
    理论上常见的控制外设,比如Usart、I2C、SPI甚至是USB等通信接口,单纯的利用CPU进行协议模拟也是可以实现的,比如51单片机经常使用I/O口模拟I2C协议通信。但这样既浪费了CPU的资源,同时实现后的性能表现往往和使用专门的硬件模块实现的效果相差甚远。从这个角度来看,各个外设控制器的存在,无疑降低了CPU的负担,解放了CPU的资源。
    
    数据搬运这一工作占用了大部分的CPU资源,成为了降低CPU的工作效率的主要原因之一。于是需要一种硬件结构分担CPU这一职能 —— DMA。
 
    从数据搬运的角度看,如果要把存储地址A的数值赋给另外一个地址上B的变量,CPU实现过程为首先读出A地址上的数据存储在一个中间变量,然后再转送到B地址的变量上。使用DMA则不需要中间变量,直接将A地址的数值传送到B地址的变量里。无疑减轻了CPU的负担,也提高了数据搬运的效率。
 
stm32中 DMA1有7个通道,DMA2有5个通道。DMA挂载的时钟为AHB总线,其时钟为72Mhz,所以可以实现高速数据搬运。
 
                                                stm32的DMA1通道一览表
 
本例实现使用CPU和DMA搬运同一组数据,通过计时,比较两者的搬运效率。
 
直接操作寄存器
 
DMA的中断状态寄存器(DMA_ISR):
 
TEIFx:通道x的传输错误标志(x = 1 … 7) (Channel x transfer error flag) 硬件设置这些位。在DMA_IFCR寄存器的相应位写入’1’可以清除这里对应的标志位。
            0:在通道x没有传输错误(TE);             1:在通道x发生了传输错误(TE)。
 
HTIFx:通道x的半传输标志(x = 1 … 7) (Channel x half transfer flag) 硬件设置这些位。在DMA_IFCR寄存器的相应位写入’1’可以清除这里对应的标志位。
            0:在通道x没有半传输事件(HT);         1:在通道x产生了半传输事件(HT)。
 
TCIFx:通道x的传输完成标志(x = 1 … 7) (Channel x transfer complete flag) 硬件设置这些位。在DMA_IFCR寄存器的相应位写入’1’可以清除这里对应的标志位。
            0:在通道x没有传输完成事件(TC);       1:在通道x产生了传输完成事件(TC)。
 
DMA_IFCR中断标志清除寄存器:
 
结构类似DMA_ISR。
 
CTEIFx:清除通道x的传输错误标志(x = 1 … 7) (Channel x transfer error clear) 这些位由软件设置和清除。     0:不起作用         1:清除DMA_ISR寄存器中的对应TEIF标志。
 
CHTIFx:清除通道x的半传输标志(x = 1 … 7) (Channel x half transfer clear) 这些位由软件设置和清除。           0:不起作用         1:清除DMA_ISR寄存器中的对应HTIF标志。
 
CTCIFx:清除通道x的传输完成标志(x = 1 … 7) (Channel x transfer complete clear) 这些位由软件设置和清除。 0:不起作用        1:清除DMA_ISR寄存器中的对应TCIF标志。
 
CGIFx:清除通道x的全局中断标志(x = 1 … 7) (Channel x global interrupt clear) 这些位由软件设置和清除。    0:不起作用         1:清除DMA_ISR寄存器中的对应的GIF、TEIF、HTIF和TCIF标志。
 
DMA通道配置寄存器(DMA_CCRx):
 
 
MEM2MEM:存储器到存储器模式 (Memory to memory mode) 该位由软件设置和清除。 0:非存储器到存储器模式; 1:启动存储器到存储器模式。
 
PL:通道优先级 (Channel priority level)  这些位由软件设置和清除。 00:低 01:中 10:高 11:最高
 
MSIZE:存储器数据宽度 (Memory size) 这些位由软件设置和清除。 00:8位 01:16位 10:32位 11:保留
 
PSIZE:外设数据宽度 (Peripheral size)  这些位由软件设置和清除。 00:8位 01:16位 10:32位 11:保留
 
MINC:存储器地址增量模式 (Memory increment mode)  该位由软件设置和清除。 0:不执行存储器地址增量操作 1:执行存储器地址增量操作
 
PINC:外设地址增量模式 (Peripheral increment mode) 该位由软件设置和清除。 0:不执行外设地址增量操作 1:执行外设地址增量操作
 
CIRC:循环模式 (Circular mode)  该位由软件设置和清除。 0:不执行循环操作 1:执行循环操作
 
DIR:数据传输方向 (Data transfer direction)   该位由软件设置和清除。 0:从外设读 1:从存储器读
 
TEIE:允许传输错误中断 (Transfer error interrupt enable)  该位由软件设置和清除。 0:禁止TE中断 0:允许TE中断
 
HTIE:允许半传输中断 (Half transfer interrupt enable) 该位由软件设置和清除。 0:禁止HT中断 0:允许HT中断
 
TCIE:允许传输完成中断 (Transfer complete interrupt enable) 该位由软件设置和清除。 0:禁止TC中断 0:允许TC中断
 
EN:通道开启 (Channel enable) 该位由软件设置和清除。 0:通道不工作 1:通道开启
 
DMA通道x传输数量寄存器(DMA_CNDTRx)(x = 1…7)
 
低16位有效。这个寄存器控制通道每次传输的数据量,数据传输数量为0至65535。该寄存器会随着传输的进行而递减,为0表示已经发送完成。
 
DMA外设地址寄存器(DMA_CPARx)
32位寄存器。外设数据寄存器的基地址,作为数据传输的源或目标。 
 
DMA存储地址寄存器(DMA_CMARx)
存储器地址[31:0],存储器地址作为数据传输的源或目标。
 
代码如下:  (system.h 和 stm32f10x_it.h 等相关代码参照 stm32 直接操作寄存器开发环境配置
User/main.c
#include <stm32f10x_lib.h>
#include "system.h"
#include "usart.h"
#include "dma.h"
#include "tim.h"
#include "string.h" #define LED1 PAout(4)
#define LED2 PAout(5)
#define LED3 PAout(6) void Gpio_Init(void); //数据源
uc32 SRC_Const_Buffer[] =
{
0x01020304,0x05060708,0x090A0B0C,0x0D0E0F10,
0x11121314,0x15161718,0x191A1B1C,0x1D1E1F20,
0x21222324,0x25262728,0x292A2B2C,0x2D2E2F30,
0x31323334,0x35363738,0x393A3B3C,0x3D3E3F40,
0x41424344,0x45464748,0x494A4B4C,0x4D4E4F50,
0x51525354,0x55565758,0x595A5B5C,0x5D5E5F60,
0x61626364,0x65666768,0x696A6B6C,0x6D6E6F70,
0x71727374,0x75767778,0x797A7B7C,0x7D7E7F80
}; //目标位置
u32 DST_Buffer[]; int main(void)
{
u8 i=;
u16 StartTime=,CPUSpendTime=,DMASpendTime=;; Rcc_Init(); //系统时钟设置 Usart1_Init(,); Tim_Init(TIM_2,,); //初始化TIM2定时器,设定重装值和分频值,计时时间为1us/次 Dma_Init(DMA1_Channel1,(u32)SRC_Const_Buffer,(u32)DST_Buffer); //初始化DMA,外设地址示例 &USART1->DR Nvic_Init(,,DMA1_Channel1_IRQChannel,); //设置抢占优先级为0,响应优先级为0,中断分组为4 Gpio_Init(); StartTime = TIM2->CNT; while(i<) //CPU搬运
{
DST_Buffer[i]=SRC_Const_Buffer[i];
i++;
} CPUSpendTime = TIM2->CNT - StartTime; printf("\r\n the CPU spend : %dus! \r\n",CPUSpendTime); if(strncmp((const char *)SRC_Const_Buffer,(const char *)DST_Buffer,) ==) //验证传输效果,判断两数组是否相同
{
printf("\r\n CPU Transmit Success! \r\n");
}else{
printf("\r\n CPU Transmit Fail! \r\n");
} i=; while(i<) //清空目标数组,准备DMA搬运
{
DST_Buffer[i]=;
i++;
} StartTime = TIM2->CNT; Dma_Enable(DMA1_Channel1,);&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//DMA搬运 while( DMA1_Channel1 -> CNDTR != ); //等待传输完成 DMASpendTime= TIM2->CNT - StartTime; printf("\r\n the DMA spend : %dus! \r\n",DMASpendTime); if(strncmp((const char *)SRC_Const_Buffer,(const char *)DST_Buffer,) ==) //验证传输效果,判断两数组是否相同
{
printf("\r\n DMA Transmit Success! \r\n");
}else{
printf("\r\n DMA Transmit Fail! \r\n");
} while();
} void Gpio_Init(void)
{
RCC->APB2ENR|=<<; //使能PORTA时钟 GPIOA->CRL&=0x0000FFFF; // PA0~3设置为浮空输入,PA4~7设置为推挽输出
GPIOA->CRL|=0x33334444; //USART1 串口I/O设置 GPIOA -> CRH&=0xFFFFF00F; //设置USART1 的Tx(PA.9)为第二功能推挽,50MHz;Rx(PA.10)为浮空输入
GPIOA -> CRH|=0x000008B0; }

User/stm32f10x_it.c

#include "stm32f10x_it.h"
#include "system.h"
#include "stdio.h" #define LED1 PAout(4)
#define LED2 PAout(5)
#define LED3 PAout(6)
#define LED4 PAout(7) void DMAChannel1_IRQHandler(void) //和启动文件有关,STM32F10x.s中 和 STM32F10x_md.s DMA中断接口函数不同
{ if( DMA1 ->ISR & (<<)) //传输完成中断
{ LED1 = ;
DMA1->IFCR |= <<; //清除传输完成中断
} if( DMA1 ->ISR & (<<)) //半传输完成中断
{ DMA1 ->IFCR |= <<; //清除半传输完成中断
} if( DMA1 ->ISR & (<<)) //传输错误中断
{
LED4 = ;
DMA1 ->IFCR |= <<; //清除传输错误中断
} DMA1 ->IFCR |= <<; //清除此通道的中断
}

Library/src/dma.c

#include <stm32f10x_lib.h>
#include "system.h"
#include "dma.h" //DMA通道初始化函数
//传输方向:存储器 -> 存储器模式 ,32位数据模式,存储器增量模式
//参数说明:
// DMA_CHx :选择DMA控制器通道,DMA1有1-7,DMA2有1-4
// P_Adress :外设地址
// M_Adress :存储器地址 void Dma_Init(DMA_Channel_TypeDef * DMA_CHx,u32 P_Address ,u32 M_Address)
{ RCC->AHBENR |= <<; DMA_CHx -> CCR &= 0xFFFF0000; //复位 DMA_CHx -> CCR |= <<; //允许传输完成中断
//DMA_CHx -> CCR |= 1<<2; //允许半传输中断
DMA_CHx -> CCR |= <<; //允许传输错误中断 读写一个保留的地址区域,将会产生DMA传输错误 //设定数据传输方向
DMA_CHx -> CCR |= <<; //设定数据传输方向 0:从外设读 1:从存储器读
DMA_CHx -> CCR |= <<; //0:不执行循环操作 1:执行循环操作 //设定地址增量
DMA_CHx -> CCR |= <<; //0:不执行外设地址增量操作 1:执行外设地址增量操作
DMA_CHx -> CCR |= <<; //0:不执行存储器地址增量操作 1:执行存储器地址增量操作 //设定外设数据宽度 S
DMA_CHx -> CCR |= <<; //外设数据宽度,由[9:8]两位控制
DMA_CHx -> CCR |= <<; //00:8位 01:16位 10:32位 11:保留 //设定存储数据宽度
DMA_CHx -> CCR |= <<; //存储器数据宽度,由[11:10]两位控制
DMA_CHx -> CCR |= <<; //00:8位 01:16位 10:32位 11:保留 //设定为中等优先级
DMA_CHx -> CCR |= <<; //通道优先级,由[13:12]两位控制
DMA_CHx -> CCR |= <<; //00:低 01:中 10:高 11:最高 DMA_CHx -> CCR |= <<; //0:非存储器到存储器模式; 1:启动存储器到存储器模式。 //必须配置好通道后配置地址
DMA_CHx -> CPAR = (u32)P_Address; //设定外设寄存器地址
DMA_CHx -> CMAR = (u32)M_Address; //设定数据存储器地址 } //DMA通道使能
//参数说明:
// DMA_CHx :选择DMA控制器通道,DMA1有1-7,DMA2有1-4
// Number :数据传输量
void Dma_Enable(DMA_Channel_TypeDef * DMA_CHx,u16 Number)
{
DMA_CHx -> CCR &= ~(<<); //关闭上一次DMA传输
DMA_CHx -> CNDTR = Number; //数据传输量
DMA_CHx -> CCR |= <<; //开始DMA传输
}

Library/inc/dma.h

#include <stm32f10x_lib.h>    

void Dma_Init(DMA_Channel_TypeDef * DMA_CHx,u32 P_Adress ,u32 M_Address);
void Dma_Enable(DMA_Channel_TypeDef * DMA_CHx,u16 Number);
直接操作寄存器输出:
 
 the CPU spend : 972us! 
 CPU Transmit Success! 
 
 the DMA spend : 5us! 
 DMA Transmit Success! 
 
库函数操作
 
mian.c
#include "stm32f10x.h"
#include "stdio.h"
#include "string.h" #define PRINTF_ON 1
#define BufferSize 32 vu16 LeftDataCounter;
vu32 Tick; uc32 SRC_Const_Buffer[BufferSize] =
{
0x01020304,0x05060708,0x090A0B0C,0x0D0E0F10,
0x11121314,0x15161718,0x191A1B1C,0x1D1E1F20,
0x21222324,0x25262728,0x292A2B2C,0x2D2E2F30,
0x31323334,0x35363738,0x393A3B3C,0x3D3E3F40,
0x41424344,0x45464748,0x494A4B4C,0x4D4E4F50,
0x51525354,0x55565758,0x595A5B5C,0x5D5E5F60,
0x61626364,0x65666768,0x696A6B6C,0x6D6E6F70,
0x71727374,0x75767778,0x797A7B7C,0x7D7E7F80
}; u32 DST_Buffer[BufferSize];
u8 i=,DMASpendTime=,CPUSpendTime=; void RCC_Configuration(void);
void GPIO_Configuration(void);
void NVIC_Configuration(void);
void USART_Configuration(void);
void DMA_Configuration(void); int main(void)
{
RCC_Configuration();
GPIO_Configuration();
NVIC_Configuration();
USART_Configuration();
DMA_Configuration(); SysTick_Config(); Tick = ;
while(i<BufferSize)
{
DST_Buffer[i]=SRC_Const_Buffer[i];
CPUSpendTime = Tick;
i++;
}
i=;
while(i<BufferSize)
{
DST_Buffer[i]=;
i++;
} Tick = ;
DMA_Cmd(DMA1_Channel6,ENABLE);
while(LeftDataCounter != ); //等待传输完成
DMASpendTime = Tick; if(strncmp((const char *)SRC_Const_Buffer,(const char *)DST_Buffer,BufferSize) ==)
{
printf("\r\n Transmit Success! \r\n");
}else{
printf("\r\n Transmit Fail! \r\n");
} printf("\r\n the CPU spend : %dus! \r\n",CPUSpendTime);
printf("\r\n the DMA spend : %dus! \r\n",DMASpendTime); } void DMA_Configuration(void)
{
DMA_InitTypeDef DMA_InitStructure; DMA_DeInit(DMA1_Channel6);
DMA_InitStructure.DMA_PeripheralBaseAddr = (u32) SRC_Const_Buffer;
DMA_InitStructure.DMA_MemoryBaseAddr = (u32) DST_Buffer;
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
DMA_InitStructure.DMA_BufferSize = BufferSize;
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Enable;
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Word;
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Word;
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal;
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Enable;
DMA_Init(DMA1_Channel6,&DMA_InitStructure); DMA_ITConfig(DMA1_Channel6,DMA_IT_TC,ENABLE);
} void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_Init(GPIOA , &GPIO_InitStructure);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOA , &GPIO_InitStructure); } void RCC_Configuration(void)
{
/* 定义枚举类型变量 HSEStartUpStatus */
ErrorStatus HSEStartUpStatus; /* 复位系统时钟设置*/
RCC_DeInit();
/* 开启HSE*/
RCC_HSEConfig(RCC_HSE_ON);
/* 等待HSE起振并稳定*/
HSEStartUpStatus = RCC_WaitForHSEStartUp();
/* 判断HSE起是否振成功,是则进入if()内部 */
if(HSEStartUpStatus == SUCCESS)
{
/* 选择HCLK(AHB)时钟源为SYSCLK 1分频 */
RCC_HCLKConfig(RCC_SYSCLK_Div1);
/* 选择PCLK2时钟源为 HCLK(AHB) 1分频 */
RCC_PCLK2Config(RCC_HCLK_Div1);
/* 选择PCLK1时钟源为 HCLK(AHB) 2分频 */
RCC_PCLK1Config(RCC_HCLK_Div2);
/* 设置FLASH延时周期数为2 */
FLASH_SetLatency(FLASH_Latency_2);
/* 使能FLASH预取缓存 */
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);
/* 选择锁相环(PLL)时钟源为HSE 1分频,倍频数为9,则PLL输出频率为 8MHz * 9 = 72MHz */
RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9);
/* 使能PLL */
RCC_PLLCmd(ENABLE);
/* 等待PLL输出稳定 */
while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);
/* 选择SYSCLK时钟源为PLL */
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);
/* 等待PLL成为SYSCLK时钟源 */
while(RCC_GetSYSCLKSource() != 0x08);
}
/* 打开APB2总线上的GPIOA时钟*/
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_USART1, ENABLE);
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //RCC_APB1PeriphClockCmd(RCC_APB1Periph_PWR|RCC_APB1Periph_BKP|RCC_APB1Periph_WWDG, ENABLE); } void USART_Configuration(void)
{
USART_InitTypeDef USART_InitStructure;
USART_ClockInitTypeDef USART_ClockInitStructure; USART_ClockInitStructure.USART_Clock = USART_Clock_Disable;
USART_ClockInitStructure.USART_CPOL = USART_CPOL_Low;
USART_ClockInitStructure.USART_CPHA = USART_CPHA_2Edge;
USART_ClockInitStructure.USART_LastBit = USART_LastBit_Disable;
USART_ClockInit(USART1 , &USART_ClockInitStructure); USART_InitStructure.USART_BaudRate = ;
USART_InitStructure.USART_WordLength = USART_WordLength_8b;
USART_InitStructure.USART_StopBits = USART_StopBits_1;
USART_InitStructure.USART_Parity = USART_Parity_No;
USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
USART_InitStructure.USART_Mode = USART_Mode_Rx|USART_Mode_Tx;
USART_Init(USART1,&USART_InitStructure); USART_Cmd(USART1,ENABLE);
} void NVIC_Configuration(void)
{
NVIC_InitTypeDef NVIC_InitStructure; NVIC_PriorityGroupConfig(NVIC_PriorityGroup_0); NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel6_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = ;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = ;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
} #if PRINTF_ON int fputc(int ch,FILE *f)
{
USART_SendData(USART1,(u8) ch);
while(USART_GetFlagStatus(USART1,USART_FLAG_TC) == RESET);
return ch;
} #endif

stm32f10x_it.c

#include "stm32f10x_it.h"
#include "stdio.h" extern vu32 Tick;
extern vu16 LeftDataCounter; void SysTick_Handler(void)
{
Tick++;
} void DMA1_Channel6_IRQHandler(void)
{ LeftDataCounter = DMA_GetCurrDataCounter(DMA1_Channel6); //获取剩余待传输数据
DMA_ClearITPendingBit(DMA1_IT_GL6);
}
库函数输出:
 
 Transmit Success! 
 the CPU spend : 68us! 
 the DMA spend : 7us! 
 

stm32 DMA数据搬运 [操作寄存器+库函数](转)的更多相关文章

  1. STM32-NVIC中断管理实现[直接操作寄存器]

    源:stm32 NVIC中断管理实现[直接操作寄存器]     cortex-m3支持256个中端,其中包含了16个内核中断,240个外部中断.stm32只有84个中断,包括16个内核中断和68个可屏 ...

  2. STM32与STM8操作寄存器的区别

    在STM8中,由于STM8寄存器较少,在头文件中定义寄存器的时候不用采取任何形式的封装,所以操作寄存器的时候直接可以用如下方式处理:PB_DDR |=0x20; 但是在STM32中,由于其寄存器实在太 ...

  3. STM32 DMA模块的配置与使用

    DMA有什么用? 直接存储器存取用来提供在外设和存储器之间或者存储器和存储器之间的高速数据传输.无须CPU的干预,通过DMA数据可以快速地移动.这就节省了CPU的资源来做其他操作. 有多少个DMA资源 ...

  4. STM32 DMA使用详解

    DMA部分我用到的相对简单,当然,可能这是新东西,我暂时还用不到它的复杂功能吧.下面用问答的形式表达我的思路. DMA有什么用? 直接存储器存取用来提供在外设和存储器之间或者存储器和存储器之间的高速数 ...

  5. (stm32学习总结)—对寄存器的理解 _

    芯片里面有什么 我们看到的 STM32 芯片是已经封装好的成品,主要由内核和片上外设组成.若与电脑类比,内核与外设就如同电脑上的 CPU 与主板.内存.显卡.硬盘的关系.STM32F103 采用的是 ...

  6. (stm32学习总结)—对寄存器的理解

    芯片里面有什么 我们看到的 STM32 芯片是已经封装好的成品,主要由内核和片上外设组成.若与电脑类比,内核与外设就如同电脑上的 CPU 与主板.内存.显卡.硬盘的关系.STM32F103 采用的是 ...

  7. STM32入门系列-存储器与寄存器介绍

    介绍两部分内容: 什么是存储器映射 什么是寄存器及寄存器映射 为了让大家对存储器与寄存器有一个更清楚的认识,并且为之后使用 C 语言来访问 STM32 寄存器内容打下基础.等明白了如何使用 C 语言封 ...

  8. stm32 DMA配置

    DMA就是将一个地址空间复制到另外一个地址空间.DMA传输方式无需CPU直接控制传输,也没有中断处理方式那样保留现场和恢复现场的过程,通过硬件为RAM与I/O设备直接传送数据,使CPU的效率大大的提高 ...

  9. LWIP network interface 即 LWIP 的 硬件 数据 接口 移植 详解 STM32 以太网数据 到达 的第二站: void ethernetif_input( void * pvParameters )

    根据 上一篇 文章 , ETH  DMA 数据中断 会 发送 一个信号量 ,我使用 全局 搜索 这个信号量 s_xSemaphore 得到 一下 几个 值 根据 这个 分析  我们找到了   数据 的 ...

随机推荐

  1. FZU 1912 Divisibility by Thirty-six(整除问题+字符串维护+贪心)

    这个整除36的与整除45的完全一样,就是被4整除的有点多,但都是两位数,所以枚举后面两位就可以了. #include <stdio.h> #include <string.h> ...

  2. Windows 2003】利用域&&组策略自动部署软件

    Windows 2003]利用域&&组策略自动部署软件 转自 http://hi.baidu.com/qu6zhi/item/4c0fa100dc768613cc34ead0 ==== ...

  3. PHPer转战Android的学习过程以及Android学习

    原文作者: eoeadmin原文地址: http://my.eoe.cn/shuhai/archive/19684.html-------------------------------------- ...

  4. 在线的代码托管平台 coding.net ===中国扩展版github

    coding.net 是国内新兴的一个项目管理平台,功能主要包括:代码托管.在线运行环境.监控代码质量,兼有一定的社交功能. 在线运行环境支持Java.Ruby.Node.js.PHP.Python. ...

  5. 为什么有时候必须添加sys.setdefaultencoding('utf-8')

    今天在尝试Python的CGI模块时遇到中文字符不能正确显示的问题,很郁闷.在网上仔细找了找,终于解决了这个问题,现在将解决方法陈述如下,以防下次失误. 页面源代码如下 #-*- coding: ut ...

  6. Redis 密码设置和登录

    Redis 一般在生产环境中,大家都不使用密码,为了确保安全,都是在防火墙上对redis端口做IP白名单的 我是个技术控,我非得了解一下密码这回事[虽然以后不会用到,呵呵] 好了,废话不多说,简单介绍 ...

  7. easyui tree使用方法

    加载数据 $('#treeul').tree({ checkbox:true, url:'initTree.aspx', onLoadSuccess:function(){ //绑定权限 $.ajax ...

  8. spider JAVA如何判断网页编码 (转载)

    原文链接 http://www.cnblogs.com/nanxin/archive/2013/03/27/2984320.html 前言 最近做一个搜索项目,需要爬取很多网站获取需要的信息.在爬取网 ...

  9. mysql中变量character_set_connection的具体作用

    如题.通常的使用中,character_set_client,character_set_connection这两个变量的值是一样的,也就是说查询不需要进行编码转换.这样看来变量character_s ...

  10. eclipse工程名出现小红叉的解决办法

    前提是eclipse工程中每个子文件都没错,工程名上却显示了小红叉. 打开[Window]->[Show View]->[General]->[Problems],看看Problem ...