热度 26
2013-3-19 16:28
1761 次阅读|
9 个评论
做MCU开发已经快将近三年了,多多少少理解了一些东西,我现在来讲讲关于如何封转模块和如何合理的分层。 从学MCU开始,就从论坛上和身边的学长听到了“模块化”一词,这一路走来,总算是摸了一些门道,或许是我实践的不够,由于工作上软硬件都要做,做软件也做的不算太多。从最初封装外设开始,用一个C文件和H文件来集成一个外设的所有功能,到现在的分层包装。其实这些都是软件工程上的东西,只不过对于我们这些学信息工程的人来说没有接触。 当然包装的越好的模块消耗不必要的资源就会越多,对于有限的MCU来说,适当即可。来看一个简单的SPI外设的例子: struct SPI_Driver { //-------controller mode -----// HANDLER01 SPI_Init; HANDLER03 RdBusyFlag; HANDLER03 WrBusyFlag; //----------io mode-----------// HANDLER02 SPI_Clk; HANDLER03 SPI_InputData; HANDLER02 SPI_OutputData; HANDLER02 SPI_Cs; }; //spi max clk fre is 1000000 hz. struct SPI_Device { struct SPI_Driver SpiDriver_mode; struct head_list SpiHead_list; UINT16 SpiNum; UINT16 SpiData; UINT8 SpiCtlMode; //spi 模式. UINT8 SpiTriggerMode; //spi trigger mode .for example up or down. UINT8 SpiSendFSM; //send fsm. UINT8 SpiRECEFSM; //receive fsm. }; //函数声明区 UINT8 ConfigSpiModule(UINT8 num, UINT32 fcy); UINT8 RegisterSpiDeviceDriver(UINT8 device_id, struct SPI_Driver *spi_operation); UINT8 SetSpiDeviceMode(UINT8 device_id, UINT8 ctl_mode, UINT8 trigger_mode); UINT8 SpiDeviceRdData(UINT8 device_id, UINT16 *data, UINT8 data_bit_num); UINT8 SpiDeviceWrData(UINT8 device_id, UINT16 *data, UINT8 data_bit_num); 这是SPI协议层的头文件,我们通过一些变量来描述该协议所具有的属性和所需要的功能,有点类似于类的包装。由于目前多数MCU都是有SPI模块的,为了构成通用,用函数指针来分别表示需要的函数接口,如果是用IO模拟SPI设备,则将用到的IO功能用函数指针来表示。 通过对SPI的理解,我们将模块包装成可以同时管理多个SPI设备,而第对他们有序的操作,由于每种SPI外设可能在读取数据位数和距离命令方面有区别,所以我列出了上述几个函数接口。 下面实现模块注册和配置函数。 UINT8 ConfigSpiModule(UINT8 num, UINT32 fcy) { _spi_num = num; _spi_fcy = fcy; _spi_delay_count = _spi_fcy / SPI_MAX_FCY; _spi_device_info = (struct SPI_Device*)My_malloc(sizeof(struct SPI_Device) * num); if(_spi_device_info == 0) return 1; My_memset((UINT8 *)_spi_device_info, sizeof(struct SPI_Device) * num, 0x00); return 0; //error return. } /* register spi device driver. */ UINT8 RegisterSpiDeviceDriver(UINT8 device_id, struct SPI_Driver *spi_operation) { if(device_id _spi_num) { _spi_device_info .SpiDriver_mode = *spi_operation; } else return 1; return 0; } /* set spi device mode. */ UINT8 SetSpiDeviceMode(UINT8 device_id, UINT8 ctl_mode, UINT8 trigger_mode) { if(device_id _spi_num) { _spi_device_info .SpiCtlMode = ctl_mode; _spi_device_info .SpiTriggerMode = trigger_mode; } else return 1; return 0; } 当使用上述三个函数接口对SPI外设进行注册和配置后,就可以独立操作每一个外设了,下面是一个SPI交换数据的DA例子。 void Reigster_TLV5620_Drv(struct SPI_Driver *TLV 5620_operation, HANDLER02 LdacPort, HANDLER02 LoadPort, UINT8 driver_mode) { _TLV5620_Info.LdacPort = LdacPort; _TLV5620_Info.LoadPort = LoadPort; RegisterSpiDeviceDriver(TLV5620_DEVICE_ID, TLV5620_operation); SetSpiDeviceMode(TLV5620_DEVICE_ID, driver_mode, DOWN_TRIGGER); TLV5620_ref_voltage = REF_VOLTAGE; //defalut value. } 这样就完成了外设DA的配置了,只需要使用这个读数据函数 SpiDeviceWrData(TLV5620_DEVICE_ID, temp, 11); //11为11位数,即通过spi传输数据位数 ,就可以将数据读到temp的变量中了。 这样一来我的这个SPI模块无论在哪里都可以使用了,实现了通用型。 当然讲了那么多的废话,其实包装模块就是抽取共性,当开始编程时,感觉到代码中有重复的东西,你可以不妨试试抽取共性,然后包装成模块,当需要使用底层的IO或者资源时,全部使用函数接口或者函数指针方式代替那部分功能,这样就实现了模块通用。 欢迎大家交流。