2010-10-25 09:03:38 +00:00
|
|
|
/** @file micro.c
|
|
|
|
* @brief STM32W108 micro specific minimal HAL functions
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* <!--(C) COPYRIGHT 2010 STMicroelectronics. All rights reserved. -->
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include PLATFORM_HEADER
|
2011-03-21 13:11:52 +01:00
|
|
|
#include BOARD_HEADER
|
2010-10-25 09:03:38 +00:00
|
|
|
#include "error.h"
|
|
|
|
#include "hal/micro/micro-common.h"
|
|
|
|
#include "hal/micro/cortexm3/micro-common.h"
|
|
|
|
#include "micro/system-timer.h"
|
|
|
|
#include "micro/adc.h"
|
2011-03-21 13:11:52 +01:00
|
|
|
#include "micro/cortexm3/memmap.h"
|
|
|
|
#include "micro/cortexm3/iap_bootloader.h"
|
2010-10-25 09:03:38 +00:00
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2011-03-21 13:11:52 +01:00
|
|
|
extern void halBoardInit(void);
|
2010-10-25 09:03:38 +00:00
|
|
|
|
|
|
|
void halInit(void)
|
|
|
|
{
|
|
|
|
//Disable the REG_EN external regulator enable signal. Out of reset this
|
|
|
|
//signal overrides PA7. By disabling it early, PA7 is reclaimed as a GPIO.
|
|
|
|
//If an external regulator is required, the following line of code should
|
|
|
|
//be deleted.
|
|
|
|
GPIO_DBGCFG &= ~GPIO_EXTREGEN;
|
|
|
|
halInternalSetRegTrim(FALSE);
|
2011-03-21 13:11:52 +01:00
|
|
|
halBoardInit();
|
2010-10-25 09:03:38 +00:00
|
|
|
halPowerUp();
|
|
|
|
halInternalCalibrateFastRc();
|
|
|
|
|
|
|
|
#ifndef DISABLE_WATCHDOG
|
|
|
|
halInternalEnableWatchDog();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
halInternalStartSystemTimer();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void halReboot(void)
|
|
|
|
{
|
|
|
|
INTERRUPTS_OFF();
|
|
|
|
|
|
|
|
|
|
|
|
//FCLK must be 6MHz to allow the SYSRESETREQ signal to cleanly
|
|
|
|
//propagate and reset the chip. Switch SYSCLK first since we need
|
|
|
|
//the cycles used by switching FCLK to guarantee the SYSCLK is
|
|
|
|
//stable and ready for SYSRESETREQ.
|
|
|
|
OSC24M_CTRL = OSC24M_CTRL_RESET; //Guarantee SYSCLK is sourced from OSCHF
|
|
|
|
CPU_CLKSEL = CPU_CLKSEL_RESET; //Guarantee FCLK is sourced from PCLK
|
|
|
|
|
|
|
|
SCS_AIRCR = (0x05FA0000 | SCS_AIRCR_SYSRESETREQ); // trigger the reset
|
|
|
|
//NOTE: SYSRESETREQ is not the same as nRESET. It will not do the debug
|
|
|
|
//pieces: DWT, ITM, FPB, vector catch, etc
|
|
|
|
}
|
|
|
|
|
|
|
|
void halPowerDown(void)
|
|
|
|
{
|
2011-03-21 13:11:52 +01:00
|
|
|
halBoardPowerDown();
|
2010-10-25 09:03:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void halPowerUp(void)
|
|
|
|
{
|
|
|
|
halInternalInitAdc();
|
|
|
|
halCommonCalibratePads();
|
|
|
|
halInternalSwitchToXtal();
|
2011-03-21 13:11:52 +01:00
|
|
|
halBoardPowerUp();
|
2010-10-25 09:03:38 +00:00
|
|
|
}
|
|
|
|
|
2013-03-15 16:14:09 +01:00
|
|
|
static uint16_t seed0 = 0xbeef;
|
|
|
|
static uint16_t seed1 = 0xface;
|
2010-10-25 09:03:38 +00:00
|
|
|
|
2013-03-15 16:14:09 +01:00
|
|
|
void halCommonSeedRandom(uint32_t seed)
|
2010-10-25 09:03:38 +00:00
|
|
|
{
|
2013-03-15 16:14:09 +01:00
|
|
|
seed0 = (uint16_t) seed;
|
2010-10-25 09:03:38 +00:00
|
|
|
if (seed0 == 0)
|
|
|
|
seed0 = 0xbeef;
|
2013-03-15 16:14:09 +01:00
|
|
|
seed1 = (uint16_t) (seed >> 16);
|
2010-10-25 09:03:38 +00:00
|
|
|
if (seed1 == 0)
|
|
|
|
seed1 = 0xface;
|
|
|
|
}
|
|
|
|
|
2013-03-15 16:14:09 +01:00
|
|
|
static uint16_t shift(uint16_t *val, uint16_t taps)
|
2010-10-25 09:03:38 +00:00
|
|
|
{
|
2013-03-15 16:14:09 +01:00
|
|
|
uint16_t newVal = *val;
|
2010-10-25 09:03:38 +00:00
|
|
|
|
|
|
|
if (newVal & 0x8000)
|
|
|
|
newVal ^= taps;
|
|
|
|
*val = newVal << 1;
|
|
|
|
return newVal;
|
|
|
|
}
|
|
|
|
|
2013-03-15 16:14:09 +01:00
|
|
|
uint16_t halCommonGetRandom(void)
|
2010-10-25 09:03:38 +00:00
|
|
|
{
|
|
|
|
return (shift(&seed0, 0x0062)
|
|
|
|
^ shift(&seed1, 0x100B));
|
|
|
|
}
|
|
|
|
|
2013-03-15 16:14:09 +01:00
|
|
|
void halCommonMemCopy(void *dest, const void *source, uint8_t bytes)
|
2010-10-25 09:03:38 +00:00
|
|
|
{
|
|
|
|
memcpy(dest, source, bytes);
|
|
|
|
}
|
|
|
|
|
2013-03-15 16:14:09 +01:00
|
|
|
int8_t halCommonMemCompare(const void *source0, const void *source1, uint8_t bytes)
|
2010-10-25 09:03:38 +00:00
|
|
|
{
|
|
|
|
return memcmp(source0, source1, bytes);
|
|
|
|
}
|
|
|
|
|
2013-03-15 16:14:09 +01:00
|
|
|
void halCommonMemSet(void *dest, uint8_t val, uint16_t bytes)
|
2010-10-25 09:03:38 +00:00
|
|
|
{
|
|
|
|
memset(dest, val, bytes);
|
|
|
|
}
|
2011-03-21 13:11:52 +01:00
|
|
|
|
|
|
|
#pragma pack(1)
|
|
|
|
typedef struct appSwitchStruct {
|
2013-03-15 16:14:09 +01:00
|
|
|
uint32_t signature;
|
|
|
|
uint8_t mode;
|
|
|
|
uint8_t channel;
|
2011-03-21 13:11:52 +01:00
|
|
|
union {
|
2013-03-15 16:14:09 +01:00
|
|
|
uint16_t panID;
|
|
|
|
uint16_t offset;
|
2011-03-21 13:11:52 +01:00
|
|
|
} param;
|
|
|
|
} appSwitchStructType;
|
|
|
|
#pragma pack()
|
|
|
|
static appSwitchStructType *appSwitch = (appSwitchStructType *) RAM_BOTTOM;
|
|
|
|
|
2013-03-15 16:14:09 +01:00
|
|
|
StStatus halBootloaderStart(uint8_t mode, uint8_t channel, uint16_t panID)
|
2011-03-21 13:11:52 +01:00
|
|
|
{
|
|
|
|
if (mode == IAP_BOOTLOADER_MODE_UART) {
|
2013-03-15 16:14:09 +01:00
|
|
|
uint8_t cut = *(volatile uint8_t *) 0x08040798;
|
2011-03-21 13:11:52 +01:00
|
|
|
if (!( (halFixedAddressTable.baseTable.type == FIXED_ADDRESS_TABLE_TYPE) &&
|
|
|
|
( ( (halFixedAddressTable.baseTable.version & FAT_MAJOR_VERSION_MASK)
|
|
|
|
== 0x0000 ) &&
|
|
|
|
(halFixedAddressTable.baseTable.version == 0x0003) //checking presence of valid version
|
|
|
|
) && (cut >= 2) && (cut <= 3)))
|
|
|
|
/* Cut not supported */
|
|
|
|
return ST_ERR_FATAL;
|
|
|
|
} else {
|
|
|
|
/* Check that OTA bootloader is at the base of the flash */
|
2013-03-15 16:14:09 +01:00
|
|
|
if (*((uint32_t *) (MFB_BOTTOM + 28)) == IAP_BOOTLOADER_APP_SWITCH_SIGNATURE) {
|
2011-03-21 13:11:52 +01:00
|
|
|
appSwitch->channel = ((channel >= 11) && (channel <= 26)) ? channel :IAP_BOOTLOADER_DEFAULT_CHANNEL;
|
|
|
|
appSwitch->param.panID = panID;
|
|
|
|
} else {
|
|
|
|
return ST_ERR_FATAL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
appSwitch->signature = IAP_BOOTLOADER_APP_SWITCH_SIGNATURE;
|
|
|
|
appSwitch->mode = mode;
|
|
|
|
halReboot();
|
|
|
|
|
|
|
|
return (mode <= IAP_BOOTLOADER_MODE_OTA) ? ST_ERR_FATAL: ST_BAD_ARGUMENT;
|
|
|
|
}
|