历史上的今天
返回首页

历史上的今天

今天是:2025年03月04日(星期二)

正在发生

2020年03月04日 | LPC2148的IAP烧写程序

2020-03-04 来源:eefocus

1 前言

继前面ADS简单程序编译分析后,得到了可以运行在ucos ii上面的相对独立编译的程序创建方法。下面就是研究如何把用户程序更新到高层的flash中(底层为ucos ii的程序)。


2 代码说明

参考LPC2148的数据手册,LPC2148支持IAP模式,即用户烧写方式。参考周立功的AN070701,iap-yingyong.pdf结合LPC21xx手册(周立功的手册对应的是LPC2300),自制了烧写函数。下面是详细的说明:


首先,参照资料,芯片通过向地址0x7FFFFFF1写入对应命令就能调用IAP烧写功能,这里不讨论芯片上电搬移程序代码的默认过程。在周立功资料里面说明了arm是如何切换到THUMB指令集工作状态的:


IAP 程序是Thumb 代码,位于地址0x7FFF FFF0。在ARM系统中实现状态转换的指令是“BX Addr”,目标地址Addr的最低位(bit0)仅来确定最终状态,实际的“目的地址= Addr & 0xFFFF FFFE”。


其实在前面的ADS简单程序编译分析已经讲解了BX汇编命令,即约定跳转指令如果最低位为1,则进入THUMB模式。这里将命令地址设计为0x7FFFFFF1则会进入THUMB模式。实际跳转到了0x7FFFFFF0。


此处需要在ADS中设置编译器的ARM C Complier里面ATPCS选项卡下面的ARM/Thumb Interworking,否则无法切换状态,也就无法执行IAP。


在ucos ii里面有如此的宏定义:


#define iap_entry(a, b)             ((void (*)())(0x7ffffff1))(a, b)

1

也就是以函数形式调用IAP的入口。


在LPC21XXX的用户手册中说明,要求跳转到0x7FFFFFF1时,r0保存命令数组的地址,r1保存返回值的数组地址。那么在C语言中,约定了函数调用第一个参数存储在r0,第二个参数存储在了r1,因此可以用下面的方法实现一个IAP调用:


static unsigned long command[5], result[3];

unsigned long IAP_PREPARE(unsigned long start, unsigned long end)

{

    OS_ENTER_CRITICAL();

    command[0] = IAP_PREPARE_SECTOR;

    command[1] = start;

    command[2] = end;

    iap_entry(command, result);

    OS_EXIT_CRITICAL();

    return result[0];

}


这个例子中,command数组是传给r0的命令数组,r1是接收结果的命令数组,在C语言中,数组名即地址。OS_ENTER_CRITICAL()是让ucos ii 禁用中断,因为在烧写过程中不能够被中断,否则会产生错误。如果没有使用ucos可以不要这些命令。在command数组中,约定了相应的数组成员对应的功能,具体可以参见LPC21XXX用户手册。


在烧写或擦除扇区之前,必须用命令让对应扇区准备好。即上面程序的代码。而ARM的flash扇区不是等长的,参照数据手册,下面的代码可以通过地址返回对应的扇区号码,其他芯片也可以照此修改qita:


/*********************************

** 函数名称: get_sector

** 函数功能: 计算地址所在的扇区号。

** 入口参数: addr 地址

** 出口参数: sector 扇区号

**********************************/

static int get_sector(unsigned int addr)

{

    if(addr <= 0x000fff) return 0;

    if(addr <= 0x001fff) return 1;

    if(addr <= 0x002fff) return 2;

    if(addr <= 0x003fff) return 3;

    if(addr <= 0x004fff) return 4;

    if(addr <= 0x005fff) return 5;

    if(addr <= 0x006fff) return 6;

    if(addr <= 0x007fff) return 7;

    if(addr <= 0x00ffff) return 8;

    if(addr <= 0x017fff) return 9;

    if(addr <= 0x01ffff) return 10;

    if(addr <= 0x027fff) return 11;

    if(addr <= 0x02ffff) return 12;

    if(addr <= 0x037fff) return 13;

    if(addr <= 0x03ffff) return 14;

    if(addr <= 0x047fff) return 15;

    if(addr <= 0x04ffff) return 16;

    if(addr <= 0x057fff) return 17;

    if(addr <= 0x05ffff) return 18;

    if(addr <= 0x067fff) return 19;

    if(addr <= 0x06ffff) return 20;

    if(addr <= 0x077fff) return 21;

    if(addr <= 0x078fff) return 22;

    if(addr <= 0x079fff) return 23;

    if(addr <= 0x07afff) return 24;

    if(addr <= 0x07bfff) return 25;

    if(addr <= 0x07cfff) return 26;

    return 0xff;

}


本设计实现了IAP的三个功能,分别是准备扇区,擦除扇区,写扇区。下面分别说明:


2.1 准备扇区函数:

/******************************************************************************************

** 函数名称: IAP_PREPARE

** 函数功能: IAP 操作缓冲区选择,代码为50。

** 入口参数: start 起始扇区

** end 终止扇区

** 出口参数: IAP 操作状态码

** IAP 返回值(result 缓冲区)

******************************************************************************************/

unsigned long IAP_PREPARE(unsigned long start, unsigned long end)

{

    OS_ENTER_CRITICAL();

    command[0] = IAP_PREPARE_SECTOR;

    command[1] = start;

    command[2] = end;

    iap_entry(command, result);

    OS_EXIT_CRITICAL();

    return result[0];

}


返回的代码如果不是0则表示出错,具体含义可以参考数据手册。


2.2 擦除扇区函数:

/******************************************************************************************

** 函数名称: EraseSector

** 函数功能: 擦除扇区,命令代码52。

** 入口参数: start 起始扇区

** end 终止扇区

** 出口参数: IAP 操作状态码

** IAP 返回值(result 缓冲区)

******************************************************************************************/

unsigned long IAP_ERASER(unsigned long start,unsigned long end)

{

    OS_ENTER_CRITICAL();

    command[0] = IAP_ERASE_SECTOR; // 设置命令字

    command[1] = start; // 设置参数

    command[2] = end;

    command[3] = IAP_FCCLK;

    iap_entry(command, result); // 调用IAP 服务程序

    OS_EXIT_CRITICAL();

    return(result[0]); // 返回状态码

}


这里的FCCLK要根据自身的芯片设置修改,单位是KHz。


2.3 写入扇区函数

/******************************************************************************************

** 函数名称: IAP_COPY

** 函数功能: 复制RAM 的数据到FLASH,命令代码51。

** 入口参数: dst 目标地址,即FLASH 起始地址,以256 字节为分界

** src 源地址,即RAM 地址,地址必须字对其

** no 复制字节个数,为256/512/1024/4096

** 出口参数: IAP 操作状态码

** IAP 返回值(result 缓冲区)

******************************************************************************************/


unsigned long IAP_COPY(unsigned long dst, unsigned long src, unsigned long no)

{

    OS_ENTER_CRITICAL();

    command[0] = IAP_COPY_RAM_TO_FLASH;

    command[1] = dst;

    command[2] = src;

    command[3] = no;

    command[4] = IAP_FCCLK;

    iap_entry(command, result);

    OS_EXIT_CRITICAL();

    return(result[0]); // 返回状态码

}


这里的FCCLK要根据自身的芯片设置修改,单位是KHz。 

此外要特别说明,写入扇区的函数不是一次写入整个扇区,而是可以分段重复写入多个位置,比如第一次写入前256个字节,然后再次继续写入1024字节,只需要遵守地址对齐要求以及写入字数为256的倍数即可。此外,在参考资料中写明,保存在内存中的待烧写内容必须是在片内RAM中,在我的应用中所有变量定义都是在RAM中没有问题,如果有片外内存或者使用了USB栈等其他片上空间的话,需要注意这一点。


3 附录

下面是完整的IAP函数文件:


#define iap_entry(a, b)             ((void (*)())(0x7ffffff1))(a, b)

#define IAP_PREPARE_SECTOR      50

#define IAP_COPY_RAM_TO_FLASH   51

#define IAP_ERASE_SECTOR        52

#define IAP_FCCLK               Fcclk / 1000;//(填写自己的主频)

/******************************************************************************************

** 函数名称: IAP_PREPARE

** 函数功能: IAP 操作缓冲区选择,代码为50。

** 入口参数: start 起始扇区

** end 终止扇区

** 出口参数: IAP 操作状态码

** IAP 返回值(result 缓冲区)

******************************************************************************************/

static unsigned long command[5], result[3];


unsigned long IAP_PREPARE(unsigned long start, unsigned long end)

{

    OS_ENTER_CRITICAL();

    command[0] = IAP_PREPARE_SECTOR;

    command[1] = start;

    command[2] = end;

    iap_entry(command, result);

    OS_EXIT_CRITICAL();

    return result[0];

}


/******************************************************************************************

** 函数名称: EraseSector

** 函数功能: 擦除扇区,命令代码52。

** 入口参数: start 起始扇区

** end 终止扇区

** 出口参数: IAP 操作状态码

** IAP 返回值(result 缓冲区)

******************************************************************************************/

unsigned long IAP_ERASER(unsigned long start,unsigned long end)

{

    OS_ENTER_CRITICAL();

    command[0] = IAP_ERASE_SECTOR; // 设置命令字

    command[1] = start; // 设置参数

    command[2] = end;

    command[3] = IAP_FCCLK;

    iap_entry(command, result); // 调用IAP 服务程序

    OS_EXIT_CRITICAL();

    return(result[0]); // 返回状态码

}



/******************************************************************************************

** 函数名称: IAP_COPY

** 函数功能: 复制RAM 的数据到FLASH,命令代码51。

** 入口参数: dst 目标地址,即FLASH 起始地址,以256 字节为分界

** src 源地址,即RAM 地址,地址必须字对其

** no 复制字节个数,为256/512/1024/4096

** 出口参数: IAP 操作状态码

** IAP 返回值(result 缓冲区)

******************************************************************************************/


unsigned long IAP_COPY(unsigned long dst, unsigned long src, unsigned long no)

{

    OS_ENTER_CRITICAL();

    command[0] = IAP_COPY_RAM_TO_FLASH;

    command[1] = dst;

    command[2] = src;

    command[3] = no;

    command[4] = IAP_FCCLK;

    iap_entry(command, result);

    OS_EXIT_CRITICAL();

    return(result[0]); // 返回状态码

}


/*********************************

** 函数名称: get_sector

** 函数功能: 计算地址所在的扇区号。

** 入口参数: addr 地址

** 出口参数: sector 扇区号

**********************************/

static int get_sector(unsigned int addr)

{

    if(addr <= 0x000fff) return 0;

    if(addr <= 0x001fff) return 1;

    if(addr <= 0x002fff) return 2;

    if(addr <= 0x003fff) return 3;

    if(addr <= 0x004fff) return 4;

    if(addr <= 0x005fff) return 5;

    if(addr <= 0x006fff) return 6;

    if(addr <= 0x007fff) return 7;

    if(addr <= 0x00ffff) return 8;

    if(addr <= 0x017fff) return 9;

    if(addr <= 0x01ffff) return 10;

    if(addr <= 0x027fff) return 11;

    if(addr <= 0x02ffff) return 12;

    if(addr <= 0x037fff) return 13;

    if(addr <= 0x03ffff) return 14;

    if(addr <= 0x047fff) return 15;

    if(addr <= 0x04ffff) return 16;

    if(addr <= 0x057fff) return 17;

    if(addr <= 0x05ffff) return 18;

    if(addr <= 0x067fff) return 19;

    if(addr <= 0x06ffff) return 20;

    if(addr <= 0x077fff) return 21;

    if(addr <= 0x078fff) return 22;

    if(addr <= 0x079fff) return 23;

    if(addr <= 0x07afff) return 24;

    if(addr <= 0x07bfff) return 25;

    if(addr <= 0x07cfff) return 26;

    return 0xff;

}

推荐阅读

史海拾趣

DIOTEC公司的发展小趣事

1973年,DIOTEC半导体股份有限公司在德国的海特尔斯海姆正式成立。公司创始人以半导体和整流产品的研发为起点,立志在电子行业中占据一席之地。在创立初期,DIOTEC就注重技术创新和品质控制,不断推出符合市场需求的高品质产品。凭借这种执着和坚持,DIOTEC逐渐在欧洲市场崭露头角。

益升华(Essentra)公司的发展小趣事

在电子产品的保护领域,益升华(Essentra)公司一直走在行业前列。多年前,公司研发团队发现市场上缺乏一种既轻便又耐用的塑料保护盖。于是,他们投入大量资源进行研发,经过数百次的试验和改进,最终成功开发出一种新型的塑料保护盖,它不仅具备优异的抗冲击性和耐磨损性,还能有效隔绝外界环境中的静电和尘埃。这一创新产品迅速获得市场的认可,为益升华(Essentra)公司带来了可观的收益。

Gems Sensors & Controls公司的发展小趣事

Gems公司的首次商业成功源自一款专为小型造船业设计的舱底开关。这款产品的推出迅速得到了船舶和工业界的认可,不仅为公司带来了可观的收入,也奠定了Gems在传感器市场的地位。随着产品的广泛应用,Gems的员工数量和设施规模不断扩大,厂房从最初的6,500平方英尺扩展至位于美国康涅狄格州普莱恩维尔市的60,000平方英尺,为公司后续的发展提供了坚实的基础。

Bellin公司的发展小趣事

Gems Sensors & Controls公司始终致力于技术创新和产品研发。数十年来,公司不断推出新型号的液位、流量和压力传感器、微型电磁阀及预装射流系统等产品,以满足客户日益多样化的需求。这些产品凭借其卓越的性能和稳定的质量,在行业内赢得了广泛的赞誉和认可,也推动了整个传感器行业的进步和发展。

Elpaq公司的发展小趣事
海芯科技(AVIA)公司的发展小趣事

随着公司业务的不断拓展和市场份额的逐步扩大,海芯科技开始注重品牌建设和市场推广。公司积极参加各类行业展会和交流活动,与业内同行和客户进行深入交流和合作。同时,公司还加大了对品牌形象的宣传力度,通过广告、宣传册等多种方式提升品牌知名度和美誉度。这些举措不仅提升了公司的市场影响力,也为公司的长期发展奠定了坚实的基础。

问答坊 | AI 解惑

单片机初学者,请进

本人是一毕业就去了深圳创业,算算时间也有差不多5年了吧,期间一直在做单片机方面的项目开发工作,现在在做嵌入式ARM的相关开发,单片机带给人的乐趣是无穷的。我个人感觉这个行业还是很有前景的。希望在校的大学生们好好利用宝贵时间多学点知识, ...…

查看全部问答>

汽车电子资料大放送~资料、文章、方案...敬请下载~

1.大众车系元件功能与检测资料下载>>https://bbs.eeworld.com.cn/thread-281-1-52.html 2.DS2705主控制器数据资料>> https://bbs.eeworld.com.cn/thread-25994-1-16.html 3.CAN总线在混和动力汽车电机控制系统中的应用>> https://bbs.eeworld ...…

查看全部问答>

mc39i tcp只能收发不了

使用外置协议栈,UDP,收发正常, TCP 3次握手过后 tcp只能接收.发送到 服务端的数据,收不到包, 发收的整个包 协议解析正常…

查看全部问答>

如何定制支持MFC的SDK

现在想把原来在WINCE5.0开发的MFC程序移植到WINCE6.0上面, 但一直不能成功.特来请教下: 我现在的环境是VS2005+WINCE6.0  之前在网上找了下, 说是WINCE6.0默认是不支持MFC的.要手动添加几个DLL文件. 现在我也把 MFC80ENU.DLL加进去了, ...…

查看全部问答>

串口调试工具发送文件和字符串的区别,香水看看

                                 我用串口调试工具来调试STM107的虚拟串口例程,发送同样字节的数据,使用串口调试工具下面的字符串输入就正常,而 ...…

查看全部问答>

单片机测量电流电压

用单片机做一个电压,电流检测装置。 (1)电压的范围:DC10-36V,要求精度1%以内。 (2)电流DC 0.1-3A,要求精度1%以内。         怎么控制这个精度问题?请各位大侠给点意见!…

查看全部问答>

16路量入,16路量出。IO不够用了,用什么芯片扩展呢??

有的人说PCA9534A,有的说输入用244,输出用373。怎么选择呢??…

查看全部问答>

求问FPGA的流水线技术

现在在做一个算法的实现,本来用了迭代结构已经实现了,当我想提高速度用流水线结构实现时,发现最高时钟频率居然比迭代结构时候还低,这个感觉跟理论对不上啊。我就是在迭代结构中需要反馈的信号都换成寄存器了,展开成很多级的。 想请问一下有遇 ...…

查看全部问答>