|
Bootloader也可以说引导程序,就像window的boot程序一样的,主要负责引导跳转,实现多功能切换,这里我们也是用这个原理实现多功能跳转,因为我们既要实现更新,又要实现正常的应用跳转。要实现这两个功能,其实很简单,引导跳转和更新固件功能可以做在一起,通过免驱的hid实现,正常的话(这里我们要检测是否有按键按下,因为资源有限,我们设置P0.1短接,这里大家不要搞混了,这里的P0.1只在bootloader里用,不会和其他冲突的)直接跳转到应用固件部分,也就是真正的编程器固件部分。
这里我们采用现成的bootloader程序修改,可以避免这块的调试测试的时间。具体你要理解怎么个运行原理,需要自己深究了,这里不做详解。
这个是主程序,其实很简单的,就是个检测按键是否按下,然后跳转相应的地址函数。
定义p0.1作为升级按键,这里我们没有用按键,采用短接插针的方式,为了方便正好又放到了isp座那里,这也算是取巧的设计了。
- void main(void)
- {
- unsigned int code *pTAdd;
- PCA0MD &= ~0x40; // Disable Watchdog timer
- pTAdd = ApplicationAddress;
- if(*pTAdd != 0xFFFF) //检测用户空间是否有程序
- {
- if(UpdataKeyDown() == 0) //检测是否满足进入条件
- {
- U_Delay();
- Jump_To_Application = (pFunction) ApplicationAddress; //进入用户程序
- Jump_To_Application();
- }
- }
- EnterBoot();
- }
复制代码 判断是否要跳到用户区,这里采用定义外部函数指针的方式
extern pFunction Jump_To_Application;
这里我们定义的用户区地址在哪呢,这个也是大家最关心的。我们为了预算以后的bootloader可能要实现其他功能,所以规划了10k的空间,这么算来我们的应用要在10k以后,那就是10240之后,十六进制就是0x2800h。
- #ifndef ApplicationAddress
- #define ApplicationAddress 0x2800
- typedef void (*pFunction)(void);
复制代码 为了和其他的hid设备区分,我们还得自定义个vip和pid
- code u_device_descriptor U_DeviceDesc =
- {
- 18, // bLength
- 0x01, // bDescriptorType
- 0x1001, // bcdUSB
- 0x00, // bDeviceClass
- 0x00, // bDeviceSubClass
- 0x00, // bDeviceProtocol
- U_EP0_PACKET_SIZE, // bMaxPacketSize0
- 0xc410, // idVendor
- 0x984e, // idProduct
- 0x0100, // bcdDevice
- 0x01, // iManufacturer
- 0x02, // iProduct
- 0x00, // iSerialNumber
- 0x01 // bNumConfigurations
- }; //end of DeviceDesc
复制代码 然后再自定义一些版权信息,方便区分
准备好这些,你就可以编译,烧写进你的340了
|
|