mirror of
https://github.com/eddyem/stm32samples.git
synced 2025-12-06 10:45:11 +03:00
247 lines
9.4 KiB
C
247 lines
9.4 KiB
C
/*
|
|
* This file is part of the multistepper project.
|
|
* Copyright 2023 Edward V. Emelianov <edward.emelianoff@gmail.com>.
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <stm32f3.h>
|
|
#include <string.h> // memcpy
|
|
#include "flash.h"
|
|
#include "hdr.h"
|
|
#include "proto.h"
|
|
#include "steppers.h"
|
|
#include "strfunc.h"
|
|
#include "usb.h"
|
|
|
|
extern const uint32_t __varsstart, _BLOCKSIZE;
|
|
|
|
static const uint32_t FLASH_blocksize = (uint32_t)&_BLOCKSIZE;
|
|
|
|
// max amount of Config records stored (will be recalculate in flashstorage_init()
|
|
static uint32_t maxCnum = 1024 / sizeof(user_conf); // can't use blocksize here
|
|
|
|
#define DEFMF {.haveencoder = 1, .donthold = 1, .eswinv = 1, .keeppos = 1}
|
|
|
|
#define USERCONF_INITIALIZER { \
|
|
.userconf_sz = sizeof(user_conf) \
|
|
,.CANspeed = 100 \
|
|
,.CANID = 0xaa \
|
|
,.microsteps = {32, 32, 32} \
|
|
,.accel = {500, 500, 500} \
|
|
,.maxspd = {2000, 2000, 2000} \
|
|
,.minspd = {20, 20, 20} \
|
|
,.maxsteps = {500000, 500000, 500000} \
|
|
,.encrev = {4000,4000,4000} \
|
|
,.encperstepmin = {17,17,17} \
|
|
,.encperstepmax = {23,23,23} \
|
|
,.motflags = {DEFMF,DEFMF,DEFMF} \
|
|
,.ESW_reaction = {ESW_IGNORE, ESW_IGNORE, ESW_IGNORE} \
|
|
}
|
|
static int erase_flash(const void*, const void*);
|
|
static int write2flash(const void*, const void*, uint32_t);
|
|
// don't write `static` here, or get error:
|
|
// 'memcpy' forming offset 8 is out of the bounds [0, 4] of object '__varsstart' with type 'uint32_t'
|
|
const user_conf *Flash_Data = (const user_conf *)(&__varsstart);
|
|
|
|
user_conf the_conf = USERCONF_INITIALIZER;
|
|
|
|
static int currentconfidx = -1; // index of current configuration
|
|
|
|
/**
|
|
* @brief binarySearch - binary search in flash for last non-empty cell
|
|
* any struct searched should have its sizeof() @ the first field!!!
|
|
* @param l - left index
|
|
* @param r - right index (should be @1 less than last index!)
|
|
* @param start - starting address
|
|
* @param stor_size - size of structure to search
|
|
* @return index of non-empty cell or -1
|
|
*/
|
|
static int binarySearch(int r, const uint8_t *start, int stor_size){
|
|
int l = 0;
|
|
while(r >= l){
|
|
int mid = l + (r - l) / 2;
|
|
const uint8_t *s = start + mid * stor_size;
|
|
if(*((const uint16_t*)s) == stor_size){
|
|
if(*((const uint16_t*)(s + stor_size)) == 0xffff){ // next is free
|
|
return mid;
|
|
}else{ // element is to the right
|
|
l = mid + 1;
|
|
}
|
|
}else{ // element is to the left
|
|
r = mid - 1;
|
|
}
|
|
}
|
|
return -1; // not found
|
|
}
|
|
|
|
/**
|
|
* @brief flashstorage_init - initialization of user conf storage
|
|
* run in once @ start
|
|
*/
|
|
void flashstorage_init(){
|
|
if(FLASH_SIZE > 0 && FLASH_SIZE < 20000){
|
|
uint32_t flsz = FLASH_SIZE * FLASH_blocksize; // size in bytes
|
|
flsz -= (uint32_t)(&__varsstart) - FLASH_BASE;
|
|
maxCnum = flsz / sizeof(user_conf);
|
|
}
|
|
// -1 if there's no data at all & flash is clear; maxnum-1 if flash is full
|
|
currentconfidx = binarySearch((int)maxCnum-2, (const uint8_t*)Flash_Data, sizeof(user_conf));
|
|
if(currentconfidx > -1){
|
|
memcpy(&the_conf, &Flash_Data[currentconfidx], sizeof(user_conf));
|
|
}
|
|
}
|
|
|
|
// store new configuration
|
|
// @return 0 if all OK
|
|
int store_userconf(){
|
|
// maxnum - 3 means that there always should be at least one empty record after last data
|
|
// for binarySearch() checking that there's nothing more after it!
|
|
if(currentconfidx > (int)maxCnum - 3){ // there's no more place
|
|
currentconfidx = 0;
|
|
if(erase_flash(Flash_Data, NULL)) return 1;
|
|
}else ++currentconfidx; // take next data position (0 - within first run after firmware flashing)
|
|
return write2flash((const void*)&Flash_Data[currentconfidx], &the_conf, sizeof(the_conf));
|
|
}
|
|
|
|
static int write2flash(const void *start, const void *wrdata, uint32_t stor_size){
|
|
int ret = 0;
|
|
if (FLASH->CR & FLASH_CR_LOCK){ // unloch flash
|
|
FLASH->KEYR = FLASH_KEY1;
|
|
FLASH->KEYR = FLASH_KEY2;
|
|
}
|
|
while (FLASH->SR & FLASH_SR_BSY);
|
|
if(FLASH->SR & FLASH_SR_WRPERR){
|
|
return 1; // write protection
|
|
}
|
|
FLASH->SR = FLASH_SR_EOP | FLASH_SR_PGERR | FLASH_SR_WRPERR; // clear all flags
|
|
FLASH->CR |= FLASH_CR_PG;
|
|
const uint16_t *data = (const uint16_t*) wrdata;
|
|
volatile uint16_t *address = (volatile uint16_t*) start;
|
|
uint32_t i, count = (stor_size + 1) / 2;
|
|
for (i = 0; i < count; ++i){
|
|
IWDG->KR = IWDG_REFRESH;
|
|
*(volatile uint16_t*)(address + i) = data[i];
|
|
while (FLASH->SR & FLASH_SR_BSY);
|
|
if(FLASH->SR & FLASH_SR_PGERR){
|
|
ret = 1; // program error - meet not 0xffff
|
|
break;
|
|
}else while (!(FLASH->SR & FLASH_SR_EOP));
|
|
FLASH->SR = FLASH_SR_EOP | FLASH_SR_PGERR | FLASH_SR_WRPERR;
|
|
}
|
|
FLASH->CR |= FLASH_CR_LOCK; // lock it back
|
|
FLASH->CR &= ~(FLASH_CR_PG);
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief erase_flash - erase N pages of flash memory
|
|
* @param start - first address
|
|
* @param end - last address (or NULL if need to erase all flash remaining)
|
|
* @return 0 if succeed
|
|
*/
|
|
static int erase_flash(const void *start, const void *end){
|
|
int ret = 0;
|
|
uint32_t nblocks = 1, flsz = 0;
|
|
if(!end){ // erase all remaining
|
|
if(FLASH_SIZE > 0 && FLASH_SIZE < 20000){
|
|
flsz = FLASH_SIZE * FLASH_blocksize; // size in bytes
|
|
flsz -= (uint32_t)start - FLASH_BASE;
|
|
}
|
|
}else{ // erase a part
|
|
flsz = (uint32_t)end - (uint32_t)start;
|
|
}
|
|
nblocks = flsz / FLASH_blocksize;
|
|
if(nblocks == 0 || nblocks >= FLASH_SIZE) return 1;
|
|
for(uint32_t i = 0; i < nblocks; ++i){
|
|
IWDG->KR = IWDG_REFRESH;
|
|
/* (1) Wait till no operation is on going */
|
|
/* (2) Clear error & EOP bits */
|
|
/* (3) Check that the Flash is unlocked */
|
|
/* (4) Perform unlock sequence */
|
|
while ((FLASH->SR & FLASH_SR_BSY) != 0){} /* (1) */
|
|
FLASH->SR = FLASH_SR_EOP | FLASH_SR_PGERR | FLASH_SR_WRPERR; /* (2) */
|
|
/* if (FLASH->SR & FLASH_SR_EOP){
|
|
FLASH->SR |= FLASH_SR_EOP;
|
|
}*/
|
|
if ((FLASH->CR & FLASH_CR_LOCK) != 0){ /* (3) */
|
|
FLASH->KEYR = FLASH_KEY1; /* (4) */
|
|
FLASH->KEYR = FLASH_KEY2;
|
|
}
|
|
/* (1) Set the PER bit in the FLASH_CR register to enable page erasing */
|
|
/* (2) Program the FLASH_AR register to select a page to erase */
|
|
/* (3) Set the STRT bit in the FLASH_CR register to start the erasing */
|
|
/* (4) Wait until the EOP flag in the FLASH_SR register set */
|
|
/* (5) Clear EOP flag by software by writing EOP at 1 */
|
|
/* (6) Reset the PER Bit to disable the page erase */
|
|
FLASH->CR |= FLASH_CR_PER; /* (1) */
|
|
FLASH->AR = (uint32_t)Flash_Data + i*FLASH_blocksize; /* (2) */
|
|
FLASH->CR |= FLASH_CR_STRT; /* (3) */
|
|
while(!(FLASH->SR & FLASH_SR_EOP));
|
|
FLASH->SR |= FLASH_SR_EOP; /* (5)*/
|
|
if(FLASH->SR & FLASH_SR_WRPERR){ /* Check Write protection error */
|
|
ret = 1;
|
|
FLASH->SR |= FLASH_SR_WRPERR; /* Clear the flag by software by writing it at 1*/
|
|
break;
|
|
}
|
|
FLASH->CR &= ~FLASH_CR_PER; /* (6) */
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
int erase_storage(){
|
|
return erase_flash(Flash_Data, NULL);
|
|
}
|
|
|
|
int fn_dumpconf(_U_ uint32_t hash, _U_ char *args){ // "dumpconf" (3271513185)
|
|
#ifdef EBUG
|
|
USB_sendstr("flashsize="); printu(FLASH_SIZE); USB_putbyte('*');
|
|
printu(FLASH_blocksize); USB_putbyte('='); printu(FLASH_SIZE*FLASH_blocksize);
|
|
newline();
|
|
#endif
|
|
USB_sendstr("userconf_addr="); printuhex((uint32_t)Flash_Data);
|
|
USB_sendstr("\nuserconf_idx="); printi(currentconfidx);
|
|
USB_sendstr("\nuserconf_sz="); printu(the_conf.userconf_sz);
|
|
USB_sendstr("\ncanspeed="); printu(the_conf.CANspeed);
|
|
USB_sendstr("\ncanid="); printu(the_conf.CANID);
|
|
// motors' data
|
|
for(int i = 0; i < MOTORSNO; ++i){
|
|
char cur = '0' + i;
|
|
#define PROPNAME(nm) do{newline(); USB_sendstr(nm); USB_putbyte(cur); USB_putbyte('=');}while(0)
|
|
PROPNAME("microsteps");
|
|
printu(the_conf.microsteps[i]);
|
|
PROPNAME("accel");
|
|
printu(the_conf.accel[i]);
|
|
PROPNAME("maxspeed");
|
|
printu(the_conf.maxspd[i]);
|
|
PROPNAME("minspeed");
|
|
printu(the_conf.minspd[i]);
|
|
PROPNAME("maxsteps");
|
|
printu(the_conf.maxsteps[i]);
|
|
PROPNAME("encperrev");
|
|
printu(the_conf.encrev[i]);
|
|
PROPNAME("encperstepmin");
|
|
printu(the_conf.encperstepmin[i]);
|
|
PROPNAME("encperstepmax");
|
|
printu(the_conf.encperstepmax[i]);
|
|
PROPNAME("motflags");
|
|
printuhex(*((uint8_t*)&the_conf.motflags[i]));
|
|
PROPNAME("eswreaction");
|
|
printu(the_conf.ESW_reaction[i]);
|
|
#undef PROPNAME
|
|
}
|
|
newline();
|
|
return RET_GOOD;
|
|
}
|