STM8单片机内部EEPROM问题

STM8单片机内部EEPROM问题,第1张

这个问题我来回答你吧
第一,是你理解错了,手册并没有错,DUL为1时解锁看程序:
do
{
FLASH_DUKR = 0x56;
FLASH_DUKR = 0xAE;
}
while((FLASH_IAPSR & 0x08) == 0);//等待解锁成功
那个While的意思:
如果FLASH_IAPSR 的BIT3位为0,那么,FLASH_IAPSR & 0x08 就会等于0,这个时候,
((FLASH_IAPSR & 0x08) == 0)这个条件就成立,那么就会循环
相反,如果FLASH_IAPSR 的BIT3位为1,那么,FLASH_IAPSR & 0x08 就会等于0x08,这个时候,
((FLASH_IAPSR & 0x08) == 0)这个条件就不会成立,那么就会退出循环
所以,DUL位为1才表示解锁,是你自己理解错了

第二个问题,所有的单片机存储方式分两种,一种是大端,另一种是小端,举例来说:

unsigned short P = (unsigned short )0x0000;
unsigned long Q = (unsigned long )0x0004;
P = 0x1234;
Q = 0x12345678
那么,如果是小端存储,
各个地址存储的内容应该如下:
地址 00 01 02 03 04 05 06 07
0x34 0x12 XXX XXX 0x78 0x56 0x34 0x12

小端比较好理解,如果是大端,各内容应该如下:
地址 00 01 02 03 04 05 06 07
0x12 0x34 XXX XXX 0x12 0x34 0x56 0x78

相信我这样举例子你应该明白了吧STM8是大端方式,

你向 unsigned int 类型的地址里写 0x55,那么被扩展成 0x0055
如果是小端方式 0x00放在高半段,也就是地址 0x4001上,而0x4000上放0x55
但是STM8是大端方式,所以 把低半段0x55放在高地址0x4001,而把高半段0x00放在低地址0x4000上

如果你想把0x55放在0x4000上,改一下函数:
void EEPROM_WriteByte(unsigned int StartAddress) //写一个字节到EEPROM地址
{
unsigned int EEPROM_Address,Address;
Address = StartAddress;
EEPROM_Address = (unsigned int )Address;
EEPROM_Unlock();
EEPROM_Address = 0x55;
while((FLASH_IAPSR & 0x04) == 0);
EEPROM_Lock();
}
把那个里面所有的 unsigned int 修改成 unsigned char

^是按位异或的符号,PD_ODR ^ 0x08是PD_ODR的值与0x08进行异或,结果值送回PD_ODR,异或是异位为1,即1异或0为1,要具体知道该程序意思,还需知道前后程序的内容。

如果会51单片机的话,那么学所有单片机步骤几乎都一样。我讲下我的经验吧,不对勿喷。我的步骤是:
1:看手册前面对单片机的功能介绍,因为我们学单片机是要做东西的,必须知道他大致功能才能了解适不适合自己的项目,比如看下ROM够不够大(能写多少程序)顺便看下可寻址大小(ROM不够就扩展);看下供电电压,一般分5V单片机和3V的,是一个范围;时钟,一般对时钟精度不高的场合不用看了,精度比较高的看下他的最大频率;看下片内外设,也就是片子本身有哪些功能,比如有些有PWM模块的,有些有数模或模数转换的,有些有温度传感器的等等,差不多了解下,如果只是为学习怎么用单片机的话,那片内外设可以先不看,那些东西就是些特殊寄存器控制的,手册也会提供接口电路,以后做项目自然就会了。
2:然后就可以试试写程序了,建议新学的话用汇编,因为汇编是基于硬件的,编写和调试过程就能让你真正理解这块芯片。先看看特殊寄存器里面的控制字,尤其是一些状态控制字,比如中断时钟什么的,这些都是必须在程序开始就初始化的;再看看一些标志位,程序跳转有时候就靠这些标志位;再看看定时器,这个所有的单片机几乎用法都一样,预置某个数然后溢出中断;再看看中断和中断优先级,一般有外部中断和定时溢出中断,看下上升沿或者下降沿触发如何设置,还有些单片机有IO口改变中断的,看了这些差不多就行了。其他的寄存器在下面写程序的时候慢慢看,需要什么看什么。
3:然后就是慢慢完善程序和硬件调试了,这个最好把他的指令集打印出来,如果你对51指令集比较熟悉的话和51的一一对应,这样调试过程中看看哪里出错,分析为什么出错,一个程序写出来(推荐写一些LED变色的程序和解码程序几乎涵盖单片机所有要用的东西),你对这款单片机就比较熟悉了,这时候,看看他的片内外设,还有其他什么功能,再慢慢往里加。对51比较熟的人,随便一款单片机有资料的话估计2个星期就能和熟练的上手了。

/
STM8S105 TIM2 TIM3输出PWM PD0 PD3 TIM3输出PWM,/#include "STM8S105Kh"
#include "STM8S105_CLOCKh"
typedef unsigned char u8;
typedef unsigned int u16;u16 value;
/
函数名 : void SystemInit(void)
功能描述: 系统初始化
函数说明: 系统全局初始化
/
void SystemInit(void)
{ SystemClock_Init();
CLK_PCKENR1 |= 0x60; //使能TIM2与TIM3与主频连接

PD_CR2 |= 0x80; //使能PD7口外部中断
}
/
函数名 : void main(void)
功能描述: 主函数
函数说明:
/
void GPIO_init(void)
{
PD_DDR = 0x1F; //配置PD端口的方向寄存器全输出
PD_CR1 = 0x1F; //设置PD为推挽输出
PB_DDR|= 0x04; //PB2输出
PB_CR1|= 0x04; //PB2推挽输出
PB_ODR|= 0x04; //开5V电源
}
void TIM2_init(void) //TIM2 CH1 工作于模式1
{
// TIM2_CCMR2= 0x60; // PWM 模式 1
TIM2_CCMR1= 0x60; // PWM 模式 1,TIM2 CH1
TIM2_CCER1= 0x03; // CC1配置为输出
//TIM2_CCER2= 0x03; // CC3使能
TIM2_ARRH = 0x03; // 配置PWM分辨率为10位,ARR=0x3FF
TIM2_ARRL = 0xFF; // PWM频率=8M/0x03FF=7820Hz
TIM2_CR1 |= 0x01; // 计数器使能,开始计数
}

void TIM3_init(void) //TIM3 CH1 ch2工作于模式2,1
{
TIM3_CCMR1= 0x70; //PWM模式2
TIM3_CCMR2= 0x60; // PWM 模式 1
TIM3_CCER1= 0x33; // CC1 CC2配置为输出,CH1 CH2

TIM3_ARRH = 0x03; // 配置PWM分辨率为10位,ARR=0x3FF
TIM3_ARRL = 0xFF; // PWM频率=8M/0x03FF=7820Hz
TIM3_CR1 = 0x01; // 计数器使能,开始计数
} void Run(void)
{ //TIM2_CCR2H = (unsigned char)(value>>8); // 更新CC2比较寄存器
//TIM2_CCR2L = (unsigned char)(value);
TIM2_CCR1=value;
//TIM2_CCR3=value;
TIM3_CCR2=value;
TIM3_CCR1=value; //注意每个通道都要设CCR
} void init_devices(void)
{
_asm("sim");
SystemInit();
GPIO_init();
TIM2_init();
TIM3_init();
_asm("rim");
} void main( void )
{
init_devices();
while(1)
Run();

} /
函数名 : @near @interrupt void TLI_IRQHandler (void)
功能描述: 中断服务程序
函数说明:
/
@near @interrupt void TLI_IRQHandler (void) {
PD_CR2 &= 0x7F; //关PD7外部中断
value+=50;
while(value>1000)
value=0; PD_CR2 |= 0x80; //使能PD7口外部中断
return;
}

stm8具有唯一id,可以利用这个id做程序加密

例如

#define ID_ENCRYPT_EOR_RESULT_ADDRESS (0x9ff0)

#define ID_ENCRYPT_ADD_RESULT_ADDRESS (0x9ff4)

void Stm8s103EncryptDemo(void)

{

      uint32 u16IdAddress;

uint16 u32EorRslt, u16AddRslt;

   

      //千万别显式的读取ID,即要把0x4865运算成隐式的,例如此例中0x4865 = (0x1194 4) + 0x215;

      //这样,别人就算破解出了你的程序,也查找不到0x4865,这样就不能轻易的软解密,这样处理后如果要软解密,

      //一定要反汇编出来进行复杂逆向分析,难度极大,代价极高,很难搞定软加密了,达到保护产品的目的。

      gU16IdAdressVar = 0x1194;

      gU16IdAdressVar <<= 2;

      u16IdAddress = (uint16)(gU16IdAdressVar + 0x215);//0x4865

 

      //读取单片机的ID,并进行运算,具体算法可以自己定,这里只用到简单的异或及和运算

      u16EorRslt = (u16IdAddress) ^ ((u16IdAddress + 1)) ^ ((u16IdAddress + 2));

      u16AddRslt = (u16IdAddress) + ((u16IdAddress + 1)) + ((u16IdAddress + 2));

      //进行对比,如果运算结果与FLASH保存的结果不一样,说明非法,运行错误代码

      if(u16EorRslt != ((uint16)ID_ENCRYPT_EOR_RESULT_ADDRESS))

      {

              while(1);//异或算法结果不正确,进行错误分支

      }

      if(u16AddRslt != ((uint32)ID_ENCRYPT_ADD_RESULT_ADDRESS))

      {

              while(1);//和算法结果不正确,进行错误分支

      }

}

1,如果板子上有外部存储器,可以先编写一个程序,利用算法把id计算得到一些值存入外部存储器,然后再烧写真正的程序,真正的程序去校验外部存储器的数据是否合法即可

2,利用板子上按键组合,或是上电按住某些键,程序在这个时候利用算法把id计算得到一些值存入程序区(stm8为EE区),程序运行时去验证程序区数据是否正确

3,轩微编程器有软件加密的功能,编程器会读芯片id,根据算法直接改写缓冲区,达到软件加密的作用

4,读出的id通过一定算法,例如异或加上一个数,得到的数据存入flash(只运行一次,运行后标志位也存入flash),下次读到这个标志位,就不运行这个程序


欢迎分享,转载请注明来源:内存溢出

原文地址:https://54852.com/yw/10540598.html

(0)
打赏 微信扫一扫微信扫一扫 支付宝扫一扫支付宝扫一扫
上一篇 2023-05-09
下一篇2023-05-09

发表评论

登录后才能评论

评论列表(0条)

    保存