了解bootloader的实现,请加QQ: 1273623966 (验证填bootloader);欢迎咨询或定制bootloader;我的博客主页www.cnblogs.com/geekygeek

今年国庆完成了4个bootloader,前面介绍了2个,都是PIC32MZ的USB bootloader, 接着介绍2个PIC24 的USB bootloader, 首先是PIC24 USB CDC bootloader。PIC24 USB CDC bootloader 是我开发给我的PIC24FJ256GB106硬件板子的。

开发环境

1. IDE: MPLABX v4.01

2. Compiler: XC16, v1.11

3. Library&Example: c:/microchip_solutions_v2013-06-15/USB/Device-CDC-Basic Demo

这个PIC24 CDC bootloader 是在MLA_v2013-06-15的USB CDC basic demo的基础上修改而成。bootloader 占用空间从0x400开始, 长度= 0x1C00。 CDC bootloader模拟UART通信,一行一行接收串口发送过来的hex原文,然后对每行的hex原文进行解析,将里面的bin数据烧写到对应的地址上。整个的逻辑实现都在我写的Boot_DoProcess()函数中,函数代码如下。

uint8_t BOOT_DoProcess(uint8_t *buffer, uint16_t byteCount)
{
uint8_t i; uint8_t bcount, recType;
DWORD_VAL pData;
uint8_t retVal = 0;
if (byteCount == 64)
{
return 1;
}
bcount = GetXbyte(buffer[LEN_NIBBLE1_INDEX],buffer[LEN_NIBBLE2_INDEX]); if (!Checksum(buffer, bcount))
{
retVal = 2;
return retVal;
} srcAddress.v[1] = GetXbyte(buffer[ADDRH_NIBBLE1_INDEX],buffer[ADDRH_NIBBLE2_INDEX]);
srcAddress.v[0] = GetXbyte(buffer[ADDRL_NIBBLE1_INDEX],buffer[ADDRL_NIBBLE2_INDEX]);
srcAddress.Val >>= 1;
recType = GetXbyte(buffer[TYPE_NIBBLE1_INDEX],buffer[TYPE_NIBBLE2_INDEX]);
switch(recType)
{
case LINEAR_ADDRESS:
srcAddress.v[3] = GetXbyte(buffer[TYPE_NIBBLE2_INDEX+1],buffer[TYPE_NIBBLE2_INDEX+2]);
srcAddress.v[2] = GetXbyte(buffer[TYPE_NIBBLE2_INDEX+3],buffer[TYPE_NIBBLE2_INDEX+4]);
retVal = 3;
break;
case DATA:
if ((srcAddress.Val >= BOOT_START_ADDRESS) && (srcAddress.Val < APPL_RESET_ADDRESS)) // boot protection, avoid to overlap
{
retVal = 4;
return retVal;
}
eraseAddress.Val = srcAddress.Val;
if ((srcAddress.Val % ERASE_BLOCK) == 0)
{
NVM_EraseBlock(eraseAddress.Val);
}
for (i=0; i < 2*bcount;)
{
pData.byte.LB = GetXbyte(buffer[TYPE_NIBBLE2_INDEX+1+i+0],buffer[TYPE_NIBBLE2_INDEX+1+i+1]);
pData.byte.HB = GetXbyte(buffer[TYPE_NIBBLE2_INDEX+1+i+2],buffer[TYPE_NIBBLE2_INDEX+1+i+3]);
pData.byte.UB = GetXbyte(buffer[TYPE_NIBBLE2_INDEX+1+i+4],buffer[TYPE_NIBBLE2_INDEX+1+i+5]);
pData.byte.MB = GetXbyte(buffer[TYPE_NIBBLE2_INDEX+1+i+6],buffer[TYPE_NIBBLE2_INDEX+1+i+7]);
unsigned int error = NVM_WriteWord(srcAddress.Val, pData.Val);
if ((error & 0x2000) > 0)
{
retVal = 5;
return retVal;
}
error = 0;
srcAddress.Val += 2;
i += 8;
}
//retVal = 1;
break;
case END:
retVal = 6;
break;
}
return retVal;
}

在main.c中ProcessIO()函数中调用Boot_DoProcess(). ProcessIO改动很大,代码如下。

void ProcessIO(void)
{
BYTE numBytesRead;
WORD status;
//Blink the LEDs according to the USB device status
BlinkUSBStatus();
// User Application USB tasks
if((USBDeviceState < CONFIGURED_STATE)||(USBSuspendControl==1)) return; if(buttonPressed)
{
if(stringPrinted == FALSE)
{
if(mUSBUSARTIsTxTrfReady())
{
putrsUSBUSART("Button Pressed -- \r\n");
stringPrinted = TRUE;
}
}
}
else
{
stringPrinted = FALSE;
} if(USBUSARTIsTxTrfReady())
{
numBytesRead = getsUSBUSART(USB_Out_Buffer,64);
if(numBytesRead != 0)
{
BYTE i;
BYTE j;
for(i=0;i<numBytesRead;i++)
{
switch(USB_Out_Buffer[i])
{
case 0x0A:
case 0x0D:
USB_In_Buffer[i] = USB_Out_Buffer[i];
if (!BOOT_Handshake)
{
BOOT_Handshake = 1;
}
else
{
if ((BOOT_RecordSOF == 1) && (BOOT_RecordEOF == 0))
{
BOOT_RecordBuffer[BOOT_RecordCounter++] = USB_Out_Buffer[i];
BOOT_RecordEOF = 1;
for (j=0; j<BOOT_RecordCounter; j++)
{
BOOT_OperationBuffer[j] = BOOT_RecordBuffer[j];
}
BOOT_OperationCounter = BOOT_RecordCounter;
BOOT_RecordLineFlag = 1;
BOOT_RecordSOF = 0;
BOOT_RecordEOF = 0;
BOOT_RecordCounter = 0;
}
}
break;
case ':':
USB_In_Buffer[i] = USB_Out_Buffer[i];
if (BOOT_Handshake)
{
if (!BOOT_RecordSOF)
{
BOOT_RecordBuffer[BOOT_RecordCounter++] = USB_Out_Buffer[i];
BOOT_RecordSOF = 1;
}
}
break;
default:
USB_In_Buffer[i] = USB_Out_Buffer[i];
if ((BOOT_Handshake == 1) && (BOOT_RecordSOF == 1))
{
BOOT_RecordBuffer[BOOT_RecordCounter++] = USB_Out_Buffer[i];
if (BOOT_RecordCounter == BOOT_RECORD_MAX)
{
BOOT_RecordEOF = 1;
for (j=0; j<BOOT_RecordCounter; j++)
{
BOOT_OperationBuffer[j] = BOOT_RecordBuffer[j];
}
BOOT_RecordLineFlag = 1;
BOOT_OperationCounter = BOOT_RecordCounter;
BOOT_RecordSOF = 0;
BOOT_RecordEOF = 0;
BOOT_RecordCounter = 0;
}
}
break;
}
}
putUSBUSART(USB_In_Buffer,numBytesRead);
if (BOOT_RecordLineFlag)
{
if (BOOT_OperationCounter == BOOT_RECORD_MAX)
{
putUSBUSART("XXXXXXXXXXXXXXXX\r\n", 18);
}
else
{
status = BOOT_DoProcess(BOOT_OperationBuffer, BOOT_OperationCounter);
if (status == 6)
{
//TODO Deinitialization
(*((void(*)(void))APPL_RESET_ADDRESS))();
}
else
{
//putUSBUSART("\r", 1);
}
}
BOOT_RecordLineFlag = 0;
}
}
if (!BOOT_Handshake)
{
BOOT_Timeout--;
if (BOOT_Timeout == 0)
{
// TODO Jump to Application
// TODO Deinitialization
(*((void(*)(void))APPL_RESET_ADDRESS))();
}
else if (BOOT_Timeout%20000 == 0)
{
putUSBUSART(".", 1);
}
}
else
{
if (!BOOT_ProgramRequired)
{
BOOT_ProgramRequired = 1;
putUSBUSART("\r\n",2);
}
}
}
CDCTxService();
} //end ProcessIO

整个CDC bootloader就完成了。合着bootloader的Linker script编译完成后,通过PICKit3烧写到硬件板子中。CDC是模拟UART通信,所以通过USB线连接到电脑,电脑可以侦测到COM口。

接着就是测试bootloader的功能了。写了个测试用的应用程序,应用程序地址是从0x2000开始到结束(地址分配参考AN1094)。应用程序合着客制的Linker script编译。测试时,我是通过超级终端发送应用程序的Hex原文。重启目标板,打开超级终端,选择CDC 模拟的COM口,配置成9600-8-none-1. 设置Line delay=40ms. 超级终端窗口出现”..."字符后,窗口中按下回车,菜单栏选择“发送文本文件”, 加载应用程序Hex, ,点击发送。然后就等着烧写完成。(上面贴的都是最终可以使用的完成函数代码,但是在实现过程中,修改几次,特别是测试过程中,发现不少问题,修改代码一一解决之后才最后完成)

以下是烧写步骤:

1. 打开超级终端

2. 重启烧录好CDC bootloader的目标板

3. 超级终端配置,选择CDC emulating后的出现的COM口,设置成9600-8-none-1, 设置Line Delay.

4. 一旦超级终端窗口出现”..."字符,立即发送回车。

5. 点击发送/发送文本文件..., 选择要发送的hex文件。

6. 等待升级完成,CDC bootloader 每接收完一行都会原文返回。

PIC24 通过USB在线升级 -- USB CDC bootloader的更多相关文章

  1. PIC32MZ 通过USB在线升级 -- USB CDC bootloader

    了解bootloader 的实现,请加QQ: 1273623966 (验证填 bootloader):欢迎咨询或定制bootloader:我的博客主页www.cnblogs.com/geekygeek ...

  2. PIC32MZ 通过USB在线升级 -- USB HID bootloader

    了解 bootloader 的实现, 请加QQ: 1273623966(验证填bootloader); 欢迎咨询或定制bootloader; 我的博客主页 www.cnblogs.com/geekyg ...

  3. PIC24 通过USB在线升级 -- USB HID bootloader

    了解bootloader的实现,请加QQ: 1273623966 (验证填bootloader):欢迎咨询或定制bootloader; 我的博客主页www.cnblogs.com/geekygeek ...

  4. PIC32MZ 通过U盘在线升级 -- USB Host bootloader

    了解bootloader的实现,请加QQ: 1273623966(验证填bootloader); 欢迎咨询或定制bootloader; 我的博客主页 www.cnblogs.com/geekygeek ...

  5. Encrypting bootloader (程序BIN文件加密及在线升级)

    了解更多关于bootloader 的C语言实现,请加我QQ: 1273623966 (验证信息请填 bootloader),欢迎咨询或定制bootloader(在线升级程序). 在上一个博客随笔,我介 ...

  6. Encrypted bootloader (程序BIN文件加密及在线升级)

    了解更多关于bootloader 的C语言实现,请加我QQ: 1273623966 (验证信息请填 bootloader),欢迎咨询或定制bootloader(在线升级程序). 在上一个博客随笔,我介 ...

  7. nrf52——DFU升级USB/UART升级方式详解(基于SDK开发例程)

    摘要:在前面的nrf52--DFU升级OTA升级方式详解(基于SDK开发例程)一文中我测试了基于蓝牙的OTA,本文将开始基于UART和USB(USB_CDC_)进行升级测试. 整体升级流程: 整个过程 ...

  8. dsp 28377在线升级 实例总结

    使用dsp品台28377d来实现在线升级的功能. 方案 : 升级程序  +  应用程序 升级程序 : 主要的目的是将上位机发送过来的应用程序数据(ccs编译生成的.bin文件)烧写到指定位置,之后在跳 ...

  9. STM32 IAP 在线升级详解(转)

    源:http://blog.csdn.net/yx_l128125/article/details/12992773 (扩展-IAP主要用于产品出厂后应用程序的更新作用,考虑到出厂时要先烧写IAP   ...

随机推荐

  1. 20150103 海南铁汉vs哈尔滨毅腾

    本文首发于『懂球帝』 这一场球赛虽然极其普通,在各位懂球帝面前或许不值得一提,但它极具历史意义,因为这是海南第一个职业联赛队伍的首场正式比赛,同时也是海南铁汉队第一次在正式比赛中与球迷们见面. 稍做一 ...

  2. CF526D Om Nom and Necklace

    嘟嘟嘟 我们可以把AB看成S,则要找的串可以写成SSSSA或者SSSSS.假设S出现了Q次,那么A出现了Q % k次,则B出现了 Q / k - Q % k次. 当ABABA是SSS的形式时,B可以为 ...

  3. python 3+djanjo 2.0.7简单学习(五)--Django投票应用

    1.编写一个简单的表单 编写的投票详细页面的模板 ("votes/detail.html") ,让它包含一个 HTML <form> 元素: <!DOCTYPE ...

  4. on-session问题

    .D:\0kecheng\bos\bosv2.0_chapter03.无条件查询. 方法1.@JSON(serialize=false)是注解排除不需要加载的实体类上,找到它的get方法,解决no-s ...

  5. stixel-net绘制指标图

    需解决问题: 1.离散点进行平滑曲线画法 https://blog.csdn.net/cdqn10086/article/details/70143616 def draw_curve(x,y,img ...

  6. 【洛谷P2746】[USACO5.3]校园网Network of Schools

    校园网Network of Schools 第一问:Tarjan缩点,搞出每一个连通块,入度为零的连通块是需要必须接受新软件副本的,统计数量即可 第二问:要让整个图构成一个环,显然要将入度为零点和出度 ...

  7. 【洛谷P3469】[POI2008]BLO-Blockade

    BLO-Blockade 题目链接 若一个点为割点:统计出每个子树的大小,两两相乘再相加, 再加上n-1,为这个点与其他点的拜访数, 因为拜访是互相的,最后再乘二即可 若一个点不是割点:只有(n-1) ...

  8. 【题解】洛谷P2532 [AHOI2012]树屋阶梯(卡特兰数+高精)

    洛谷P2532:https://www.luogu.org/problemnew/show/P2532 思路 来自Sooke大佬的推导: https://www.luogu.org/blog/Sook ...

  9. AngularJS 控制器属性

    <!DOCTYPE html><html><head><meta http-equiv="Content-Type" content=&q ...

  10. 排序算法 JavaScript

    一.冒泡排序 算法介绍: 1.比较相邻的两个元素,如果前一个比后一个大,则交换位置. 2.第一轮把最大的元素放到了最后面. 3.由于每次排序最后一个都是最大的,所以之后按照步骤1排序最后一个元素不用比 ...