您的位置:首页 > 运维架构

STM8S---选项字节(Option Byte)写操作之IO复用

2017-07-18 15:39 281 查看
转自:http://blog.csdn.net/FreeApe/article/details/47008033


功能实现目标

  通过对选项字节的写操作来实现TIM2的CH3通道的PWM输出IO复用,可以设置为PA3或者PD2输出。 
通过STVP方式操作链接


选项字节

  选项字节包括芯片硬件特性的配置和存储器的保护信息,这些字节保存在存储器中一个专用的块内。除了ROP(读出保护)字节,每个选项字节必须被保存两次,一个是通常的格式(OPTx)和一个用来备份互补格式的(NOPTx)。选项字节可以通过应用程序在IAP模式下修改,但是ROP选项只能在ICP模式(通过SWIM)下被修改。有关SWIM编程过程的内容可以参考STM8S闪存编程手册(PM0051)和STM8 SWIM通信协议和调试模块用户手册(UM0470)。 

  不同的芯片的选项字节大小不一样,具体的可以参考芯片数据手册。如用STM8S103F3来举例,选项字节如下:



  STM8S103F对于20脚封装的产品的复用功能重定义位:



  由此可知我们要将OPT2字节中的AFR1位进行写操作,通过写0,则端口A3复用功能TIM2_CH3,通过写1,端口D2复用功能TIM2_CH3。接下来通过程序实现这个功能,可以修改AFR1的值来看PWM输出是否切换了管脚,如果能,则是实现了写操作。 

  选项字节存放在EEPROM中,所以可以通过读写EEPROM一样的操作方式来修改选项字节。应用程序可以直接向目标地址进行写操作。地址从上面的图中我们已经知道了:0x4803,0x4804。寄存器的配置可以查阅参考手册(RM0016)。 

  相关寄存器操作:






测试程序实现

  注意:实现程序擦写Option Bytes时,不能运行应用程序,否则会出现错误!不过还是觉得这点麻烦啊,还不如用STVP来擦写,要是能够放在应用程序中共存来擦写就好了,用程序实现复用就这么麻烦吗?希望能探索找到好的方法,最后只找到了个不靠谱的,就是在擦写后加上延时,但是这个时管用时不管用。还是再查阅资料看看是怎么回事?



  但是用STVP擦写时又遇到了这个错误:
Error : Error on Option Bytes (complementary bytes). Reprogram Option Bytes of device
Error : < OPTION BYTE verifying failed.


  用STVP来擦写Option Bytes了,先将ROP设置为ON,然后再擦写Option Bytes,会出现两个提示框,选择是(Y),再之后又将ROP设置为OFF,再次擦写Option Bytes,则又可以用STVD通过stlink来烧写程序并仿真了。







测试程序:

/* MAIN.C file
Functons    :   操作option byte字节,设置IO复用,来修改TIM2的CH3通道PWM输出管脚PA3 or PD2
Date        :   2015年7月22日
Author      :   yicm
Notes       :
*/

#include<stm8s003f3p.h>

void CLK_init(void)
{
CLK_ICKR |= 0X01;           //使能内部高速时钟 HSI
CLK_CKDIVR = 0x08;          //16M内部RC经2分频后系统时钟为8M
while(!(CLK_ICKR&0x02));    //HSI准备就绪
CLK_SWR=0xe1;               //HSI为主时钟源
}

void Init_GPIO(void)
{
/*设置为推挽输出,PD2接了LED灯*/
PD_DDR |= 0X04;             //设置PD2端口为输出模式
PD_CR1 |= 0X04;             //设置PD2端口为推挽输出模式
PD_CR2 &= 0XFD;

PA_DDR |= 0X08;             //设置PA3端口为输出模式
PA_CR1 |= 0X08;             //设置PA3端口为推挽输出模式
PA_CR2 |= 0XF7;
}

void Init_Tim2(void)
{
TIM2_CCMR3 |=   0X70;   //设置定时器2三通道(PD2)输出比较三模式
TIM2_CCMR3 |= 0X04;     //输出比较3预装载使能

TIM2_CCER2 |= 0x03;     //通道3使能,低电平有效,配置为输出

// 初始化时钟分频器为1,即计数器的时钟频率为Fmaster=8M/64=0.125MHZ
TIM2_PSCR = 0X07;
//初始化自动装载寄存器,决定PWM 方波的频率,Fpwm=0.125M/62500=2HZ
TIM2_ARRH = 62500/256;
TIM2_ARRL = 62500%256;
//初始化比较寄存器,决定PWM 方波的占空比:5000/10000 = 50%
TIM2_CCR3H = 31250/256;
TIM2_CCR3L = 31250%256;

//启动计数;更新中断失能
TIM2_CR1 |= 0x81;
//TIM2_IER |= 0x00;
}

void Write_Option_Byte(void)
{
unsigned char opt[6] = {0,0,0x00,0,0,0};

/*解锁Flash*/
do
{
FLASH_DUKR = 0xAE;
FLASH_DUKR = 0x56;
}
while(!(FLASH_IAPSR & 0X08));

/*对选项字节进行写操作使能*/
FLASH_CR2 = 0X80;
/*互补控制寄存器*/
FLASH_NCR2 = 0X7F;
/*写操作,0x02:PD2。0x00:PA3*/
*((unsigned char *)0x4800) = opt[0];

*((unsigned char *)0x4801) = opt[1];
*((unsigned char *)0x4802) = ~opt[1];

*((unsigned char *)0x4803) = opt[2];
*((unsigned char *)0x4804) = ~opt[2];

*((unsigned char *)0x4805) = opt[3];
*((unsigned char *)0x4806) = ~opt[3];

*((unsigned char *)0x4807) = opt[4];
*((unsigned char *)0x4808) = ~opt[0];

*((unsigned char *)0x4809) = opt[5];
*((unsigned char *)0x480A) = ~opt[5];

/*等待写结束*/
while(!(FLASH_IAPSR & 0x04));
}

main()
{
int i;

Write_Option_Byte();    //运行程序时,屏蔽
for(i=0;i<10000;++i);   //延时效果,有时加上延时,能够使擦写和应用程序同时不屏蔽也能管用

CLK_init();             //擦写时屏蔽,否则下次stlink仿真时会出错
Init_GPIO();            //擦写时屏蔽,否则下次stlink仿真时会出错
Init_Tim2();            //擦写时屏蔽,否则下次stlink仿真时会出错
while (1);
}


用程序改写选项字参考,转自:http://www.stmcu.org/module/forum/thread-606022-1-1.html

先看外设时钟门控寄存器配置没有,我有stm8s103的选项字节配置你参考下:
void FLASH_Init(void)
{
//解锁数据 EEPROM
FLASH_DUKR = 0xae;
delay_us(1);
FLASH_DUKR = 0x56;
delay_us(1);
//判断解锁是否成功,即判断 FLASH_IAPSR 寄存器的 DUL 位是否为 1
while(!(FLASH_IAPSR & 0x08));
delay_us(1);
//使能选项字节写操作
FLASH_CR2 = 0x80;
delay_us(1);
FLASH_NCR2 = 0x7f;
delay_us(1);
*((unsigned char *)0x4803) = 0x01;
delay_us(1);
*((unsigned char *)0x4804) = 0xfe;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: