本程序主要利用I2C串行總線,實現AT24Cxx系列EEPROM存儲器(此處是AT24C02)的讀寫,將數據寫入,再讀出發送至串口
可利用EEPROM存儲器數據斷電不消失性質存儲一些配置數據等。
主程序
/*******************************************************************************
*
* 軟件功能: I2C讀寫AT24Cxx系列EEPROM存儲器
*
*******************************************************************************/
#include "stm32f10x.h"
#include
#include "delay.h"
#include "I2C.h"
#include "AT24Cxx.h"
void RCC_Configuration(void);
void GPIO_Configuration(void);
void USART1_Configuration(void);
void Uart1_PutChar(u8 ch);
void Uart1_PutString(u8* buf , u8 len);
int fputc(int ch, FILE *f);
/*************************************************
函數: int main(void)
功能: main主函數
參數: 無
返回: 無
**************************************************/
int main(void)
{
u16 tempdata=0;
u16 i=0;
RCC_Configuration();
GPIO_Configuration();
delay_init(72);
USART1_Configuration();
I2C_Configuration();
delay_ms(1);
for(i=0;i<255;i++)
{
AT24Cxx_WriteOneByte(i,i);
}
for(i=0;i<255;i++)
{
tempdata=AT24Cxx_ReadOneByte(i);
printf("%x ",tempdata);
}
//AT24Cxx_WriteTwoByte(0,0x1234);
//tempdata=AT24Cxx_ReadTwoByte(0);
// printf("兩個字節 dt=%x\n",tempdata);
while(1);
}
/*************************************************
函數: void RCC_Configuration(void)
功能: 復位和時鐘控制 配置
參數: 無
返回: 無
**************************************************/
void RCC_Configuration(void)
{
ErrorStatus HSEStartUpStatus; //定義外部高速晶體啟動狀態枚舉變量
RCC_DeInit(); //復位RCC外部設備寄存器到默認值
RCC_HSEConfig(RCC_HSE_ON); //打開外部高速晶振
HSEStartUpStatus = RCC_WaitForHSEStartUp(); //等待外部高速時鐘準備好
if(HSEStartUpStatus == SUCCESS) //外部高速時鐘已經準別好
{
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //開啟FLASH預讀緩沖功能,加速FLASH的讀取。所有程序中必須的用法.位置:RCC初始化子函數里面,時鐘起振之后
FLASH_SetLatency(FLASH_Latency_2); //flash操作的延時
RCC_HCLKConfig(RCC_SYSCLK_Div1); //配置AHB(HCLK)時鐘等于==SYSCLK
RCC_PCLK2Config(RCC_HCLK_Div1); //配置APB2(PCLK2)鐘==AHB時鐘
RCC_PCLK1Config(RCC_HCLK_Div2); //配置APB1(PCLK1)鐘==AHB1/2時鐘
RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_9); //配置PLL時鐘 == 外部高速晶體時鐘 * 9 = 72MHz
RCC_PLLCmd(ENABLE); //使能PLL時鐘
while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET) //等待PLL時鐘就緒
{
}
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //配置系統時鐘 = PLL時鐘
while(RCC_GetSYSCLKSource() != 0x08) //檢查PLL時鐘是否作為系統時鐘
{
}
}
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_GPIOB | RCC_APB2Periph_USART1 | RCC_APB2Periph_AFIO, ENABLE); //允許 GPIOA、GPIOB、USART1、AFIO時鐘
}
/*************************************************
函數: void GPIO_Configuration(void)
功能: GPIO配置
參數: 無
返回: 無
**************************************************/
void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure; //定義GPIO初始化結構體
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //復合推挽輸出
GPIO_Init(GPIOA, &GPIO_InitStructure); //PA9串口輸出
}
/*******************************************************************************
函數名:USART1_Configuration
輸 入:
輸 出:
功能說明:
初始化串口硬件設備,啟用中斷
配置步驟:
(1)打開GPIO和USART1的時鐘
(2)設置USART1兩個管腳GPIO模式
(3)配置USART1數據格式、波特率等參數
(4)使能USART1接收中斷功能
(5)最后使能USART1功能
*/
void USART1_Configuration(void) //串口配置 詳見《STM32的函數說明(中文).pdf》P346
{
USART_InitTypeDef USART_InitStructure;
USART_InitStructure.USART_BaudRate=9600; //波特率為9600
USART_InitStructure.USART_WordLength=USART_WordLength_8b; //數據位為8
USART_InitStructure.USART_StopBits=USART_StopBits_1; //在幀結尾傳輸 1 個停止位
USART_InitStructure.USART_Parity=USART_Parity_No; //校驗模式:奇偶失能
USART_InitStructure.USART_HardwareFlowControl=USART_HardwareFlowControl_None; //硬件流控制失能
USART_InitStructure.USART_Mode=USART_Mode_Tx | USART_Mode_Rx; //USART_Mode 指定了使能或者失能發送和接收模式:發送使能|接收失能
USART_Init(USART1, &USART_InitStructure); //初始化配置
USART_Cmd(USART1,ENABLE); //使能或者失能 USART 外設
USART_ClearFlag(USART1, USART_FLAG_TC);//清除傳輸完成標志位,否則可能會丟失第1個字節的數據.USART_FLAG_TC為發送完成標志位
}
//發送一個字符
void Uart1_PutChar(u8 ch)
{
USART_SendData(USART1, (u8) ch);
while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET);//等待發送完成
}
//發送一個字符串 Input : buf為發送數據的地址 , len為發送字符的個數
void Uart1_PutString(u8* buf , u8 len)
{
u8 i;
for(i=0;i { Uart1_PutChar(*(buf++)); } } int fputc(int ch, FILE *f) { Uart1_PutChar((u8)ch); //此處為自定義函數,參見串口中斷通信,請勿盲目復制 return (ch); } I2C.h #ifndef __I2C_H #define __I2C_H #include "stm32f10x.h" //如果移植程序時只要改一下三個地方就行了 #define I2C_SCL GPIO_Pin_6 #define I2C_SDA GPIO_Pin_7 #define GPIO_I2C GPIOB #define I2C_SCL_H GPIO_SetBits(GPIO_I2C,I2C_SCL) #define I2C_SCL_L GPIO_ResetBits(GPIO_I2C,I2C_SCL) #define I2C_SDA_H GPIO_SetBits(GPIO_I2C,I2C_SDA) #define I2C_SDA_L GPIO_ResetBits(GPIO_I2C,I2C_SDA) void I2C_Configuration(void); void I2C_SDA_OUT(void); void I2C_SDA_IN(void); void I2C_Start(void); void I2C_Stop(void); void I2C_Ack(void); void I2C_NAck(void); u8 I2C_Wait_Ack(void); void I2C_Send_Byte(u8 txd); u8 I2C_Read_Byte(u8 ack); #endif I2C.c #include "delay.h" #include "I2C.h" void I2C_Configuration(void) { GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin=I2C_SCL|I2C_SDA; GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP; GPIO_Init(GPIOB,&GPIO_InitStructure); I2C_SCL_H; I2C_SDA_H; } void I2C_SDA_OUT(void) { GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin=I2C_SDA; GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP; GPIO_Init(GPIOB,&GPIO_InitStructure); } void I2C_SDA_IN(void) { GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin=I2C_SDA; GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU; GPIO_Init(GPIOB,&GPIO_InitStructure); } //產生起始信號 void I2C_Start(void) { I2C_SDA_OUT(); I2C_SDA_H; I2C_SCL_H; delay_us(5); I2C_SDA_L; delay_us(6); I2C_SCL_L; } //產生停止信號 void I2C_Stop(void) { I2C_SDA_OUT(); I2C_SCL_L; I2C_SDA_L; I2C_SCL_H; delay_us(6); I2C_SDA_H; delay_us(6); } //主機產生應答信號ACK void I2C_Ack(void) { I2C_SCL_L; I2C_SDA_OUT(); I2C_SDA_L; delay_us(2); I2C_SCL_H; delay_us(5); I2C_SCL_L; } //主機不產生應答信號NACK void I2C_NAck(void) { I2C_SCL_L; I2C_SDA_OUT(); I2C_SDA_H; delay_us(2); I2C_SCL_H; delay_us(5); I2C_SCL_L; } //等待從機應答信號 //返回值:1 接收應答失敗 // 0 接收應答成功 u8 I2C_Wait_Ack(void) { u8 tempTime=0; I2C_SDA_IN(); I2C_SDA_H; delay_us(1); I2C_SCL_H; delay_us(1); while(GPIO_ReadInputDataBit(GPIO_I2C,I2C_SDA)) { tempTime++; if(tempTime>250) { I2C_Stop(); return 1; } } I2C_SCL_L; return 0; } //I2C 發送一個字節 void I2C_Send_Byte(u8 txd) { u8 i=0; I2C_SDA_OUT(); I2C_SCL_L;//拉低時鐘開始數據傳輸 for(i=0;i<8;i++) { if((txd&0x80)>0) //0x80 1000 0000 I2C_SDA_H; else I2C_SDA_L; txd<<=1; I2C_SCL_H; delay_us(2); //發送數據 I2C_SCL_L; delay_us(2); } } //I2C 讀取一個字節 u8 I2C_Read_Byte(u8 ack) { u8 i=0,receive=0; I2C_SDA_IN(); for(i=0;i<8;i++) { I2C_SCL_L; delay_us(2); I2C_SCL_H; receive<<=1; if(GPIO_ReadInputDataBit(GPIO_I2C,I2C_SDA)) receive++; delay_us(1); } if(ack==0) I2C_NAck(); else I2C_Ack(); return receive; } AT24Cxx.h #ifndef _AT24Cxx_H #define _AT24Cxx_H #include "stm32f10x.h" #include "I2C.h" #include "delay.h" #define AT24C01 127 #define AT24C02 255 #define AT24C04 511 #define AT24C08 1023 #define AT24C16 2047 #define AT24C32 4095 #define AT24C64 8191 #define AT24C128 16383 #define AT24C256 32767 #define EE_TYPE AT24C02 u8 AT24Cxx_ReadOneByte(u16 addr); u16 AT24Cxx_ReadTwoByte(u16 addr); void AT24Cxx_WriteOneByte(u16 addr,u8 dt); void AT24Cxx_WriteTwoByte(u16 addr,u16 dt); #endif AT24Cxx.c #include "AT24Cxx.h" u8 AT24Cxx_ReadOneByte(u16 addr) { u8 temp=0; I2C_Start(); if(EE_TYPE>AT24C16) { I2C_Send_Byte(0xA0); I2C_Wait_Ack(); I2C_Send_Byte(addr>>8); //發送數據地址高位 } else { I2C_Send_Byte(0xA0+((addr/256)<<1));//器件地址+數據地址 } I2C_Wait_Ack(); I2C_Send_Byte(addr%256);//雙字節是數據地址低位 //單字節是數據地址低位 I2C_Wait_Ack(); I2C_Start(); I2C_Send_Byte(0xA1); I2C_Wait_Ack(); temp=I2C_Read_Byte(0); // 0 代表 NACK I2C_Stop(); return temp; } u16 AT24Cxx_ReadTwoByte(u16 addr) { u16 temp=0; I2C_Start(); if(EE_TYPE>AT24C16) { I2C_Send_Byte(0xA0); I2C_Wait_Ack(); I2C_Send_Byte(addr>>8); //發送數據地址高位 } else { I2C_Send_Byte(0xA0+((addr/256)<<1));//器件地址+數據地址 } I2C_Wait_Ack(); I2C_Send_Byte(addr%256);//雙字節是數據地址低位 //單字節是數據地址低位 I2C_Wait_Ack(); I2C_Start(); I2C_Send_Byte(0xA1); I2C_Wait_Ack(); temp=I2C_Read_Byte(1); // 1 代表 ACK temp<<=8; temp|=I2C_Read_Byte(0); // 0 代表 NACK I2C_Stop(); return temp; } void AT24Cxx_WriteOneByte(u16 addr,u8 dt) { I2C_Start(); if(EE_TYPE>AT24C16) { I2C_Send_Byte(0xA0); I2C_Wait_Ack(); I2C_Send_Byte(addr>>8); //發送數據地址高位 } else { I2C_Send_Byte(0xA0+((addr/256)<<1));//器件地址+數據地址 } I2C_Wait_Ack(); I2C_Send_Byte(addr%256);//雙字節是數據地址低位 //單字節是數據地址低位 I2C_Wait_Ack(); I2C_Send_Byte(dt); I2C_Wait_Ack(); I2C_Stop(); delay_ms(10); } void AT24Cxx_WriteTwoByte(u16 addr,u16 dt) { I2C_Start(); if(EE_TYPE>AT24C16) { I2C_Send_Byte(0xA0); I2C_Wait_Ack(); I2C_Send_Byte(addr>>8); //發送數據地址高位 } else { I2C_Send_Byte(0xA0+((addr/256)<<1));//器件地址+數據地址 } I2C_Wait_Ack(); I2C_Send_Byte(addr%256);//雙字節是數據地址低位 //單字節是數據地址低位 I2C_Wait_Ack(); I2C_Send_Byte(dt>>8); I2C_Wait_Ack(); I2C_Send_Byte(dt&0xFF); I2C_Wait_Ack(); I2C_Stop(); delay_ms(10); }
上一篇:s3c2440_LCD控制器設置及代碼詳解
下一篇:STM32單片機(12) 紅外信號接收解碼(外部中斷)
推薦閱讀
史海拾趣
隨著技術實力的增強,Dae Ryung Electronic Co Ltd公司開始積極拓展市場。公司制定了國際化戰略,逐步進入國際市場。通過參加國際展會、與海外企業建立合作關系等方式,公司成功打開了海外市場的大門。同時,公司還針對不同地區的市場需求,推出定制化的產品和服務,進一步提升了市場競爭力。
隨著全球環保意識的不斷提高,電子行業對綠色環保和可持續發展的要求也日益嚴格。CCI積極響應這一趨勢,將綠色環保理念融入公司的戰略規劃和日常運營中。
公司不僅采用了環保材料和綠色生產工藝來降低產品對環境的影響,還通過優化生產流程和資源利用等方式降低能耗和排放。此外,CCI還積極參與行業環保組織和公益活動,推動整個行業向更加綠色、環保的方向發展。
Deutsch公司自創立之初,就注重技術創新和產品研發。在早期階段,公司成功開發出一款具有突破性的電子元件,這款元件以其高性能和低成本迅速在市場上獲得了認可。隨著技術的不斷進步,Deutsch公司不斷推出創新產品,滿足日益增長的市場需求。這些技術突破和產品創新不僅提升了公司的競爭力,也推動了整個電子行業的發展。
為了進一步加強在全球市場的地位,C&K在2011年進行了一次重要的收購。這次收購不僅擴大了公司的業務范圍,還提升了其設計和制造能力。通過這次收購,C&K得以在全球范圍內提供更廣泛的產品和服務,進一步鞏固了其在電子行業中的領先地位。
在激烈的市場競爭中,品牌建設和形象提升對于企業的發展至關重要。Comclok Inc深知品牌建設的重要性,從產品設計、生產到銷售服務,都注重塑造公司的品牌形象。公司注重產品的品質和用戶體驗,不斷提升產品的性能和穩定性。同時,Comclok Inc還積極參加各類行業展會和交流活動,展示公司的技術實力和產品優勢,提升了公司在行業內的知名度和影響力。
隨著電子行業的快速發展,技術創新和合作成為了企業發展的重要驅動力。Cal Crystal Lab Inc深知這一點,積極尋求與其他企業的技術合作。通過與國內外知名高校和研究機構的深入合作,公司不斷引進新技術、新工藝,提升了產品的技術含量和附加值。同時,Cal Crystal Lab Inc還與其他企業建立了戰略合作關系,共同開發新產品、拓展新市場,實現了共贏發展。
你知道嗎?ARM每年可派出40億個處理器,全球超過四分之一的電子產品使用了ARM技術,其中,以微控制器為主的嵌入式應用是未來10年增長最快的領域。 伴隨著授權數量的增長,支持ARM微控制器內核的生態系統也在蓬勃發展,包括工具、中間件和 ...… 查看全部問答∨ |
Xilinx platform cable usb(Xilinx 下載線) 1.目標器件VCC兼容 1.5V--5V 2.可配置所有Xilinx器件 3.支持iMPACT 和ChipScope 4.支持JTAG和Slave Serial配置模式 5.目標器件下載時鐘可選,最高可達24 MHz 6.LED狀態指示燈指示目標器件電 ...… 查看全部問答∨ |
我學過c,java, 理解還可以!我想學嵌入式!曾經學過單片機,但基礎很差!有人告訴我 先從單片機學起! 我需要 買板子嗎?是不是 需要兩塊板子!一個先學單片機,一個學arm! 有沒有 一塊板子,都可以學的! 我還需要學哪些方面! ...… 查看全部問答∨ |
|
想在PXA270板子上使用內置USB設備,涉及驅動結構 USBFN<--->USBD|HCD 如果我想用外接USB設備,就要涉及USB ROOT HUB的驅動, 請問USB ROOT HUB驅動是在上述結構的什么位置? USBFN里面對內置和外置USB設備處理有沒有不同? 小弟初學,表述 ...… 查看全部問答∨ |
【免費下載】2008年全國11城市STM32技術研討會演講稿已經上線 請訪問ST中文網頁:http://www.stmicroelectronics.com.cn/ 在最左邊“產品”欄目點擊“微控制器” 在“大中華地區相關信息”欄目點擊“中國大陸地區” 在“STM32  ...… 查看全部問答∨ |
微型打印機是5v供電的,26線雙排針型插座 打印機的輸入信號DB0~DB7,用3.3v到5v的轉換芯片4245 那么打印機的輸出信號ACK,BUSY想直接接到149行么… 查看全部問答∨ |