mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
ff7f8ae2f0
RIOT supports two distinct families of the MSP430: The [MSP430 x1xx] MCU family and the [MSP430 F2xx/G2xx] MCU family. For both incompatible MCU families the code was located in the msp430fxyz folder, resulting in case of the UART driver in particularly bizarre code looking roughly like this: #ifndef UART_USE_USCI /* implementation of x1xx peripheral ... */ #else /* implementation of F2xx/G2xx peripheral ... */ #endif /* zero shared code between both variants */ This splits the peripheral drivers for USCI and USART serial IP blocks into separate files and relocates everything in cpu/msp430, similar to how cpu/stm32 is organized. [MSP430 x1xx]: https://www.ti.com/lit/ug/slau049f/slau049f.pdf [MSP430 F2xx/G2xx]: https://www.ti.com/lit/ug/slau144k/slau144k.pdf
119 lines
2.7 KiB
C
119 lines
2.7 KiB
C
/*
|
|
* Copyright (C) 2014 INRIA
|
|
* 2017 Freie Universität Berlin
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup cpu_msp430
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Implementation of the peripheral flashpage interface
|
|
*
|
|
* @author Oliver Hahm <oliver.hahm@inria.fr>
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <assert.h>
|
|
|
|
#include "cpu.h"
|
|
#include "irq.h"
|
|
#include "periph/flashpage.h"
|
|
|
|
/**
|
|
* @brief Memory markers, defined in the linker script
|
|
* @{
|
|
*/
|
|
extern uint32_t _end_fw;
|
|
extern uint32_t _erom;
|
|
|
|
static inline int _unlock(void)
|
|
{
|
|
int state;
|
|
state = irq_disable();
|
|
FCTL3 = FWKEY;
|
|
while (FCTL3 & BUSY) {}
|
|
return state;
|
|
}
|
|
|
|
static inline void _lock(int state)
|
|
{
|
|
FCTL3 = (FWKEY | LOCK);
|
|
irq_restore(state);
|
|
}
|
|
|
|
static inline void _erase(uint16_t *page_addr)
|
|
{
|
|
/* disable interrupts and unlock flash */
|
|
int state = _unlock();
|
|
|
|
/* erase page */
|
|
FCTL1 = (FWKEY | ERASE);
|
|
*page_addr = 0;
|
|
while (FCTL3 & BUSY) {}
|
|
|
|
/* lock flash and re-enable interrupts */
|
|
_lock(state);
|
|
}
|
|
|
|
void flashpage_erase(unsigned page)
|
|
{
|
|
assert((unsigned) page < FLASHPAGE_NUMOF);
|
|
|
|
uint16_t *page_addr = (uint16_t *)flashpage_addr(page);
|
|
|
|
/* erase page */
|
|
_erase(page_addr);
|
|
}
|
|
|
|
void flashpage_write(void *target_addr, const void *data, size_t len)
|
|
{
|
|
/* assert multiples of FLASHPAGE_WRITE_BLOCK_SIZE are written and no less of
|
|
that length. */
|
|
assert(!(len % FLASHPAGE_WRITE_BLOCK_SIZE));
|
|
|
|
/* ensure writes are aligned */
|
|
assert(!(((unsigned)target_addr % FLASHPAGE_WRITE_BLOCK_ALIGNMENT) ||
|
|
((unsigned)data % FLASHPAGE_WRITE_BLOCK_ALIGNMENT)));
|
|
|
|
/* ensure the length doesn't exceed the actual flash size */
|
|
assert(((unsigned)target_addr + len) <
|
|
(CPU_FLASH_BASE + (FLASHPAGE_SIZE * FLASHPAGE_NUMOF) - 1));
|
|
|
|
uint8_t *page_addr = target_addr;
|
|
const uint8_t *data_addr = data;
|
|
|
|
/* disable interrupts and unlock flash */
|
|
int state = _unlock();
|
|
|
|
/* enable write access, and write*/
|
|
FCTL1 = (FWKEY | WRT);
|
|
for (unsigned i = 0; i < len; i++) {
|
|
*(page_addr++) = *(data_addr++);
|
|
while (!(FCTL3 & WAIT)) {}
|
|
}
|
|
/* disable write access */
|
|
FCTL1 = (FWKEY);
|
|
|
|
/* lock flash and re-enable interrupts */
|
|
_lock(state);
|
|
}
|
|
|
|
unsigned flashpage_first_free(void)
|
|
{
|
|
return flashpage_page(&_end_fw) + 1;
|
|
}
|
|
|
|
/* MSP430 cpu's last page holds the interrupt vector, so flashpage_last_free
|
|
is the one before last */
|
|
unsigned flashpage_last_free(void)
|
|
{
|
|
return flashpage_page(&_erom) - 1;
|
|
}
|