[DemoCode下载] 使用USB及I2C 去存取EEPROM

[复制链接]
 楼主| a_ziliu 发表于 2016-12-26 10:43 | 显示全部楼层 |阅读模式
使用USB 及I2C去存取EEPROM
EC_NUC230_240_HID_EEPROM.zip (1.13 MB, 下载次数: 35)
643757107 发表于 2016-12-26 14:47 | 显示全部楼层
EEPROM一般是I2C 的接口吧,如果用USB去弄,那意思是用这个芯片的USB接口来跟计算机通信了?
langziwuliao 发表于 2016-12-26 16:49 | 显示全部楼层
对的
598330983 发表于 2016-12-26 16:50 | 显示全部楼层
一般常用的EEPROM都是I2C接口。
nobleliom 发表于 2016-12-30 20:08 | 显示全部楼层
usb接口操作eeprom,这个怎么实现呢?好像无法操作吧,eeprom用的是I2C
yiyigirl2014 发表于 2016-12-31 10:03 | 显示全部楼层
帮楼主贴出来,方便大家阅读
  1. /******************************************************************************
  2. * [url=home.php?mod=space&uid=288409]@file[/url]     main.c
  3. * [url=home.php?mod=space&uid=247401]@brief[/url]    NANO100 series USBD driver source file
  4. * [url=home.php?mod=space&uid=895143]@version[/url]  2.0.0
  5. * [url=home.php?mod=space&uid=212281]@date[/url]     22, March, 2013
  6. *
  7. * @note
  8. * Copyright (C) 2013 Nuvoton Technology Corp. All rights reserved.
  9. ******************************************************************************/
  10. #include <stdio.h>
  11. #include "NUC230_240.h"
  12. #include "hid_transfer.h"
  13. #define I2C_EEPROM I2C1
  14. #define I2C_EEPROM_IRQn I2C1_IRQn

  15. void EnableCLKO(uint32_t u32ClkSrc, uint32_t u32ClkDiv)
  16. {
  17.     /* CLKO = clock source / 2^(u32ClkDiv + 1) */
  18.     CLK->FRQDIV = CLK_FRQDIV_DIVIDER_EN_Msk | u32ClkDiv;

  19.     /* Enable CLKO clock source */
  20.     CLK->APBCLK |= CLK_APBCLK_FDIV_EN_Msk;

  21.     /* Select CLKO clock source */
  22.     CLK->CLKSEL2 = (CLK->CLKSEL2 & (~CLK_CLKSEL2_FRQDIV_S_Msk)) | u32ClkSrc;
  23. }

  24. /*--------------------------------------------------------------------------*/
  25. void SYS_Init(void)
  26. {

  27.     /*---------------------------------------------------------------------------------------------------------*/
  28.     /* Init System Clock                                                                                       */
  29.     /*---------------------------------------------------------------------------------------------------------*/

  30.     /* Enable Internal RC 22.1184MHz clock */
  31.     CLK_EnableXtalRC(CLK_PWRCON_OSC22M_EN_Msk);

  32.     /* Waiting for Internal RC clock ready */
  33.     CLK_WaitClockReady(CLK_CLKSTATUS_OSC22M_STB_Msk);

  34.     /* Switch HCLK clock source to Internal RC and HCLK source divide 1 */
  35.     CLK_SetHCLK(CLK_CLKSEL0_HCLK_S_HIRC, CLK_CLKDIV_HCLK(1));

  36.     /* Enable external XTAL 12MHz clock */
  37.     CLK_EnableXtalRC(CLK_PWRCON_XTL12M_EN_Msk);

  38.     /* Waiting for external XTAL clock ready */
  39.     CLK_WaitClockReady(CLK_CLKSTATUS_XTL12M_STB_Msk);

  40.     /* Set Flash Access Delay */
  41.     FMC->FATCON |= FMC_FATCON_FOMSEL1_Msk | FMC_FATCON_FOMSEL0_Msk;

  42.     /* Set core clock */
  43.     CLK_SetCoreClock(72000000);

  44.     /* Enable module clock */
  45.     CLK_EnableModuleClock(UART0_MODULE);
  46.     CLK_EnableModuleClock(USBD_MODULE);

  47.     /* Select module clock source */
  48.     CLK_SetModuleClock(UART0_MODULE, CLK_CLKSEL1_UART_S_HXT, CLK_CLKDIV_UART(1));
  49.     CLK_SetModuleClock(USBD_MODULE, 0, CLK_CLKDIV_USB(3));


  50.     /*---------------------------------------------------------------------------------------------------------*/
  51.     /* Init I/O Multi-function                                                                                 */
  52.     /*---------------------------------------------------------------------------------------------------------*/

  53.     /* Set GPB multi-function pins for UART0 RXD and TXD, and Clock Output */
  54.     SYS->GPB_MFP |= (SYS_GPB_MFP_PB0_UART0_RXD | SYS_GPB_MFP_PB1_UART0_TXD | SYS_GPB_MFP_PB8_CLKO);
  55.     SYS->ALT_MFP |=  SYS_ALT_MFP_PB8_CLKO;

  56.     /* Enable CLKO(PB8) for monitor HCLK. CLKO = HCLK/8 Hz*/
  57.     EnableCLKO((2 << CLK_CLKSEL2_FRQDIV_S_Pos), 2);
  58. }


  59. void UART0_Init(void)
  60. {
  61.     /*---------------------------------------------------------------------------------------------------------*/
  62.     /* Init UART                                                                                               */
  63.     /*---------------------------------------------------------------------------------------------------------*/
  64.     /* Reset IP */
  65.     SYS->IPRSTC2 |=  SYS_IPRSTC2_UART0_RST_Msk;
  66.     SYS->IPRSTC2 &= ~SYS_IPRSTC2_UART0_RST_Msk;

  67.     /* Configure UART0 and set UART0 Baudrate */
  68.     UART0->BAUD = UART_BAUD_MODE2 | UART_BAUD_MODE2_DIVIDER(__HXT, 115200);
  69.     UART0->LCR = UART_WORD_LEN_8 | UART_PARITY_NONE | UART_STOP_BIT_1;
  70. }
  71. extern uint8_t volatile g_u8EP3Ready;

  72. typedef __packed struct {
  73.     uint8_t u8PID;
  74.     uint8_t u8Command;
  75.     uint8_t u8Size;
  76.     uint8_t u8Buffer[61];       
  77. } CMD_T;

  78. CMD_T gCmd;
  79. #define I2C_SET_ADDRESS 0x01
  80. #define I2C_WRITE 0x02
  81. #define I2C_READ 0x03

  82. uint8_t g_u8DeviceAddr;
  83. uint8_t g_u8RxData;
  84. uint8_t g_u8DataLen;
  85. volatile uint8_t g_u8EndFlag = 0;

  86. typedef void (*I2C_FUNC)(uint32_t u32Status);
  87. static I2C_FUNC s_I2CHandlerFn = NULL;

  88. /*---------------------------------------------------------------------------------------------------------*/
  89. /*  I2C IRQ Handler                                                                                       */
  90. /*---------------------------------------------------------------------------------------------------------*/
  91. void I2C1_IRQHandler(void)
  92. {
  93.     uint32_t u32Status;

  94.     u32Status = I2C_EEPROM->I2CSTATUS;

  95.     if (I2C_EEPROM->I2CTOC & I2C_I2CTOC_TIF_Msk)
  96.     {
  97.         /* Clear I2C Timeout Flag */
  98.         I2C_EEPROM->I2CTOC |= I2C_I2CTOC_TIF_Msk;                  
  99.     }   
  100.     else
  101.     {
  102.         if (s_I2CHandlerFn != NULL)
  103.             s_I2CHandlerFn(u32Status);
  104.     }
  105. }

  106. /*---------------------------------------------------------------------------------------------------------*/
  107. /*  I2C Initial Function                                                                                       */
  108. /*---------------------------------------------------------------------------------------------------------*/
  109. __INLINE void I2C_PIN_Init(void)
  110. {
  111.                 /* Set GPA10,11 multi-function pins for I2C1 SDA and SCL */
  112.                 SYS->GPA_MFP |= SYS_GPA_MFP_PA10_I2C1_SDA|SYS_GPA_MFP_PA11_I2C1_SCL;
  113.        
  114.           /* Enable I2C1 clock */        
  115.     CLK->APBCLK |= CLK_APBCLK_I2C1_EN_Msk;
  116.        
  117.     /* Reset I2C1 */
  118.     SYS->IPRSTC2 |=  SYS_IPRSTC2_I2C1_RST_Msk;
  119.     SYS->IPRSTC2 &= ~SYS_IPRSTC2_I2C1_RST_Msk;       
  120. }

  121. void I2C_EEPROM_Init(void)
  122. {
  123.                 I2C_PIN_Init();
  124.     /* Enable I2C Controller */
  125.     //I2C_EEPROM->I2CON |= I2C_I2CON_ENS1_Msk;
  126.    
  127.     /* I2C clock divider, I2C Bus Clock = PCLK / (4*120) = 100kHz */
  128.     //I2C_EEPROM->I2CLK = I2C_I2CLK_DIV4(u8Divider);
  129.     I2C_Open(I2C_EEPROM, 100000);
  130.     /* Enable I2C interrupt and set corresponding NVIC bit */
  131.     I2C_EEPROM->I2CON |= I2C_I2CON_EI_Msk;
  132.     NVIC_EnableIRQ(I2C_EEPROM_IRQn);
  133. }


  134. /*---------------------------------------------------------------------------------------------------------*/
  135. /*  I2C Rx Callback Function                                                                               */
  136. /*---------------------------------------------------------------------------------------------------------*/
  137. void I2C_MasterRx(uint32_t u32Status)
  138. {
  139.         #if 0
  140.     if (u32Status == 0x08)                      /* START has been transmitted and prepare SLA+W */
  141.     {
  142.         I2C_EEPROM->I2CDAT = g_u8DeviceAddr << 1;     /* Write SLA+W to Register I2CDAT */
  143.         I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_SI);
  144.     }   
  145.     else if (u32Status == 0x18)                 /* SLA+W has been transmitted and ACK has been received */
  146.     {
  147.         I2C_EEPROM->I2CDAT = gCmd.u8Buffer[g_u8DataLen++];
  148.         I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_SI);
  149.     }
  150.     else if (u32Status == 0x20)                 /* SLA+W has been transmitted and NACK has been received */
  151.     {
  152.         I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_STA_STO_SI);
  153.     }
  154.     else if (u32Status == 0x28)                 /* DATA has been transmitted and ACK has been received */
  155.     {
  156.         if (g_u8DataLen != 2)
  157.         {
  158.             I2C_EEPROM->I2CDAT = gCmd.u8Buffer[g_u8DataLen++];
  159.             I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_SI);
  160.         }
  161.         else
  162.         {
  163.             I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_STA_SI);
  164.         }      
  165.     }
  166.                 #endif
  167.     if (u32Status == 0x08)                 /* Repeat START has been transmitted and prepare SLA+R */
  168.     {
  169.         I2C_EEPROM->I2CDAT = ((g_u8DeviceAddr << 1) | 0x01);   /* Write SLA+R to Register I2CDAT */
  170.         I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_SI);
  171.     }
  172.     else if (u32Status == 0x40)                 /* SLA+R has been transmitted and ACK has been received */
  173.     {
  174.         I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_SI|I2C_I2CON_SI_AA);
  175.     }      
  176.     else if (u32Status == 0x50)                 /* DATA has been received and NACK has been returned */
  177.     {
  178.                           if(g_u8DataLen!= gCmd.u8Size-1)
  179.                                 {
  180.         gCmd.u8Buffer[g_u8DataLen++]= I2C_EEPROM->I2CDAT;
  181.         I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_SI|I2C_I2CON_SI_AA);
  182.         }else{
  183.                                 gCmd.u8Buffer[g_u8DataLen++]= I2C_EEPROM->I2CDAT;
  184.         I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_SI);
  185.                                 }
  186.     }
  187.                 else if (u32Status == 0x58)                 /* DATA has been received and NACK has been returned */
  188.     {
  189.         g_u8RxData = I2C_EEPROM->I2CDAT;
  190.         I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_STO_SI);
  191.         g_u8EndFlag = 1;
  192.     }
  193.     else
  194.     {
  195.         /* TO DO */
  196.         printf("Status 0x%x is NOT processed\n", u32Status);
  197.                                 while(1);
  198.     }           
  199. }

  200. /*---------------------------------------------------------------------------------------------------------*/
  201. /*  I2C Tx Callback Function                                                                               */
  202. /*---------------------------------------------------------------------------------------------------------*/
  203. void I2C_MasterTx(uint32_t u32Status)
  204. {
  205.     if (u32Status == 0x08)                      /* START has been transmitted */
  206.     {
  207.         I2C_EEPROM->I2CDAT = g_u8DeviceAddr << 1;     /* Write SLA+W to Register I2CDAT */
  208.         I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_SI);
  209.     }   
  210.     else if (u32Status == 0x18)                 /* SLA+W has been transmitted and ACK has been received */
  211.     {
  212.         I2C_EEPROM->I2CDAT = gCmd.u8Buffer[g_u8DataLen++];
  213.         I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_SI);
  214.     }
  215.     else if (u32Status == 0x20)                 /* SLA+W has been transmitted and NACK has been received */
  216.     {
  217.         I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_STA_STO_SI);
  218.     }   
  219.     else if (u32Status == 0x28)                 /* DATA has been transmitted and ACK has been received */
  220.     {
  221.         if (g_u8DataLen != gCmd.u8Size)
  222.         {
  223.             I2C_EEPROM->I2CDAT = gCmd.u8Buffer[g_u8DataLen++];
  224.             I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_SI);
  225.         }
  226.         else
  227.         {
  228.             I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_STO_SI);
  229.             g_u8EndFlag = 1;
  230.         }      
  231.     }
  232.     else
  233.     {
  234.         /* TO DO */
  235.         //printf("Status 0x%x is NOT processed\n", u32Status);
  236.                                 while(1);
  237.     }      
  238. }

  239. void I2C_EEPROM_Write(void)
  240. {
  241.                 g_u8DataLen = 0;
  242.                 g_u8EndFlag = 0;
  243.                
  244.                 /* I2C function to write data to slave */
  245.                 s_I2CHandlerFn = (I2C_FUNC)I2C_MasterTx;
  246.                
  247.                 /* I2C as master sends START signal */
  248.                 I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_STA);

  249.                 /* Wait I2C Tx Finish */
  250.                 while (g_u8EndFlag == 0);       
  251. }

  252. /*---------------------------------------------------------------------------------------------------------*/
  253. /*  Read I2C EEPROM                                                                                        */
  254. /*---------------------------------------------------------------------------------------------------------*/
  255. uint8_t I2C_EEPROM_Read(void)
  256. {
  257.                 g_u8DataLen = 0;
  258.                 g_u8EndFlag = 0;
  259.                
  260.                 /* I2C function to write data to slave */
  261.                 s_I2CHandlerFn = (I2C_FUNC)I2C_MasterRx;
  262.                
  263.                 /* I2C as master sends START signal */
  264.                 I2C_SET_CONTROL_REG(I2C_EEPROM, I2C_I2CON_STA);

  265.                 /* Wait I2C Tx Finish */
  266.                 while (g_u8EndFlag == 0);       
  267.        
  268.                 return g_u8RxData;
  269. }


  270. /*---------------------------------------------------------------------------------------------------------*/
  271. /*  Main Function                                                                                          */
  272. /*---------------------------------------------------------------------------------------------------------*/
  273. int32_t main(void)
  274. {
  275.          uint8_t *ptrOut, *ptrIn;
  276.     /*
  277.         This sample code demonstrate how to use HID interface to transfer data
  278.         between PC and USB device.
  279.         A demo window tool are also included in "WindowsTool" directory with this
  280.         sample code. User can use it to test data transfer with this sample code.

  281.     */

  282.     /* Unlock write-protected registers */
  283.     SYS_UnlockReg();

  284.     /* Init system and multi-funcition I/O */
  285.     SYS_Init();

  286.     /* Init UART for debug message */
  287.     UART0_Init();
  288.           I2C_EEPROM_Init();
  289.     printf("NuMicro USB HID Transfer Sample\n");

  290.     /* Open USB controller */
  291.     USBD_Open(&gsInfo, HID_ClassRequest, NULL);

  292.     /*Init Endpoint configuration for HID */
  293.     HID_Init();

  294.     /* Start USB device */
  295.     USBD_Start();

  296.     /* Enable USB device interrupt */
  297.     NVIC_EnableIRQ(USBD_IRQn);

  298.     while(1)
  299.                 {               
  300.                 if(g_u8EP3Ready==1)
  301.                 {
  302.                         g_u8EP3Ready=0;
  303.                             
  304.     /* Interrupt OUT */
  305.       ptrOut = (uint8_t *)(USBD_BUF_BASE + USBD_GET_EP_BUF_ADDR(EP3));
  306.                         ptrIn = (uint8_t *)(USBD_BUF_BASE + USBD_GET_EP_BUF_ADDR(EP2));
  307.      
  308.       USBD_MemCopy((uint8_t *)&gCmd, ptrOut, 64);
  309.                        
  310.                         if(gCmd.u8Command==I2C_SET_ADDRESS)
  311.                         {
  312.                                 g_u8DeviceAddr=gCmd.u8Size;
  313.                                 printf("i2c address 0x%x\n\r",gCmd.u8Size);
  314.                         }
  315.                        
  316.                         if(gCmd.u8Command==I2C_WRITE)
  317.                         {
  318.                                 printf("i2c write\n\r");
  319.        I2C_EEPROM_Write();
  320.                         }
  321.                        
  322.                  if(gCmd.u8Command==I2C_READ)
  323.                         {
  324.                                 printf("i2c read\n\r");
  325.        I2C_EEPROM_Read();
  326.                         }
  327.                         gCmd.u8PID+=1;
  328.                        
  329.                        
  330.                         USBD_MemCopy(ptrIn, (void *)&gCmd, 64);
  331.                         USBD_SET_PAYLOAD_LEN(EP3, EP3_MAX_PKT_SIZE);                       
  332.       USBD_SET_PAYLOAD_LEN(EP2, EP2_MAX_PKT_SIZE);
  333.                   }
  334.                 }
  335. }



  336. /*** (C) COPYRIGHT 2013 Nuvoton Technology Corp. ***/


huangcunxiake 发表于 2016-12-31 10:29 | 显示全部楼层
原来是I2C 读写外部的存储器
捉虫天师 发表于 2016-12-31 20:32 | 显示全部楼层
我认为单片机最复杂的其实是时钟系统
您需要登录后才可以回帖 登录 | 注册

本版积分规则

100

主题

310

帖子

6

粉丝
快速回复 在线客服 返回列表 返回顶部