cnxh 发表于 2017-10-20 09:44:04

mega1284p,boot写入地址错位

执行以下函数 p_address 页地址为0时,调试写入的地址是0X8000,p_address 页地址为0x80时,调试写入的动作是0Xc000,
计算了以下,好像对应的高位地址是1,即页地址0,对应的flash地址为0x10000,而同样的程序,mega644却正确
//执行操作,擦除(code=0x03)和写入(code=0x05)一个Flash页
void boot_page_ew(uint16 p_address,uint8 code)
{         
            asm("mov r30,r17");
            asm("mov r31,r16");
            asm("out 0x3b,r18");                  //将页地址放入Z寄存器和RAMPZ的Bit0中
            SPMCSR = code;                        //寄存器SPMCSR中为操作码
            asm("spm");                      //对指定Flash页进行spm操作      
}   

cnxh 发表于 2017-10-20 09:58:49

是否还有个设置是选择,低64k,或高64k的

cnxh 发表于 2017-10-20 10:01:13

现在我觉的是这个页地址是8位的,0对应的是高64k开始,0xff对应的高64k的最后一页

cnxh 发表于 2017-10-21 04:52:08

p_address高位地址手动设置成0x100,R30高位页面地址也是01了,写入的还是flash,0x8000地址,和页面地址0x0000一样,等于高位地址不是用R30控制的,手册上说是r30,是不是高位地址是另一个寄存器,

cnxh 发表于 2017-10-21 09:15:29

自己解决void boot_page_ew(uint16 p_address,uint8 code) 传送类型uint16 p_address,,改成longp_address, 就可以了,这样p_address =0,写能定位到flash地址的开始0000处了,这个问题搞了我2天,为什么每次发布问题,网站都没人帮我解决,每次最后都是自己解决

andy 发表于 2017-10-21 10:37:07

cnxh 发表于 2017-10-21 09:15
自己解决void boot_page_ew(uint16 p_address,uint8 code) 传送类型uint16 p_address,,改成longp_address ...

为什么你自己的问题,一定要别人来帮你解决你的问题,真奇怪?

cnxh 发表于 2017-10-21 22:29:39

本帖最后由 cnxh 于 2017-10-21 22:32 编辑

你没看我说的吗,自己搞了2天搞不起,才上网寻求帮助的,你难道碰到问题,自己解决不了,不寻求帮助的吗,况且自己解决,也把解决方法发布,让别人少走弯路,资源共享,不是只索取,不付出

cnxh 发表于 2017-10-22 10:04:39

本帖最后由 cnxh 于 2017-10-22 10:05 编辑

原来那个不是根本原因,低6k擦除正常了,高位64k却覆盖了低64k,原因是这一句
asm("out 0x3b,r18");                  //将页地址放入Z寄存器和RAMPZ的Bit0中
手册上说RAMPZ bit0才是页面选择,这里R18的值并不能传送所需要的高位值,网上找的全部是使用r18传送
这句才正常asm("out 0x3b,r17");//r17传送的高位地址,设置到页面选择
现在高64k和低64k已全部正常

t3486784401 发表于 2017-10-24 15:37:02

1. AVR 是小端处理器(LE),所以你 r17=>r30, r16=>r31 的过程存在逻辑问题,除非你用的编译器坚持反过来BE传参数。
   正确的小端拷贝应该是 r17=>r31, r16=>R30 (对应两个mov指令),或者 {r17:r16}=>{r31:r30} (对应一个movw指令)。

2. 这个涉及 C-汇编 混编的操作,还是不要直接写寄存器比较合适,容易出错(比如之前的r18,实际把 code=>rampz)。
   我在 ICCAVR 里边允许汇编引用 C 符号的,例如 asm("movwr30, %p_address");
   至于 unsigned long 的参数传递,ICCAVR里边用到了软件栈(Y指针),看来很是因编译器而异。

3. 如果你没意识到 r17/r16 写颠倒了,即使 8 楼的结论,估计也只能是在 r16=r17 (例如零地址)时不出错而已。

cnxh 发表于 2017-10-27 05:54:23

谢谢,asm("mov r30,r17");
asm("mov r31,r16"); ,我这样传送才是全部正确的,网上的都是反过来的,高低位才是颠倒的

appfan 发表于 2017-11-8 14:53:41

大家都开始c语言与汇编混合编程了!厉害!!!

cnxh 发表于 2017-11-8 23:24:42

appfan 发表于 2017-11-8 14:53
大家都开始c语言与汇编混合编程了!厉害!!!

boot只能汇编,因为有指令严格时间限制,和必须直接定位到指定寄存器

appfan 发表于 2017-11-9 10:33:33

弱弱的问一句,atmega8,atmega16,atmega32它们的boot写入汇编指令相同么???

cnxh 发表于 2017-11-10 20:46:02

只用过644pa和1284p,估计相同
页: [1]
查看完整版本: mega1284p,boot写入地址错位