2010-04-28 08:47:36 +00:00
|
|
|
/*
|
2012-08-16 17:55:41 +00:00
|
|
|
* (C) Copyright 2010-2012
|
2010-04-28 08:47:36 +00:00
|
|
|
* Stefan Roese, DENX Software Engineering, sr@denx.de.
|
|
|
|
*
|
2013-07-08 07:37:19 +00:00
|
|
|
* SPDX-License-Identifier: GPL-2.0+
|
2010-04-28 08:47:36 +00:00
|
|
|
*/
|
|
|
|
|
2012-08-16 17:55:41 +00:00
|
|
|
#include <bootcount.h>
|
|
|
|
#include <linux/compiler.h>
|
2010-04-28 08:47:36 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Only override CONFIG_SYS_BOOTCOUNT_ADDR if not already defined. This
|
|
|
|
* way, some boards can define it directly in their config header.
|
|
|
|
*/
|
|
|
|
#if !defined(CONFIG_SYS_BOOTCOUNT_ADDR)
|
|
|
|
|
|
|
|
#if defined(CONFIG_MPC5xxx)
|
|
|
|
#define CONFIG_SYS_BOOTCOUNT_ADDR (MPC5XXX_CDM_BRDCRMB)
|
|
|
|
#define CONFIG_SYS_BOOTCOUNT_SINGLEWORD
|
|
|
|
#endif /* defined(CONFIG_MPC5xxx) */
|
|
|
|
|
2010-05-20 14:09:34 +00:00
|
|
|
#if defined(CONFIG_MPC512X)
|
|
|
|
#define CONFIG_SYS_BOOTCOUNT_ADDR (&((immap_t *)CONFIG_SYS_IMMR)->clk.bcr)
|
|
|
|
#define CONFIG_SYS_BOOTCOUNT_SINGLEWORD
|
|
|
|
#endif /* defined(CONFIG_MPC512X) */
|
|
|
|
|
2010-04-28 08:47:36 +00:00
|
|
|
#if defined(CONFIG_8xx)
|
|
|
|
#define CONFIG_SYS_BOOTCOUNT_ADDR (((immap_t *)CONFIG_SYS_IMMR)->im_cpm.cp_dpmem + \
|
|
|
|
CPM_BOOTCOUNT_ADDR)
|
|
|
|
#endif /* defined(CONFIG_8xx) */
|
|
|
|
|
|
|
|
#if defined(CONFIG_MPC8260)
|
|
|
|
#include <asm/cpm_8260.h>
|
|
|
|
#define CONFIG_SYS_BOOTCOUNT_ADDR (CONFIG_SYS_IMMR + CPM_BOOTCOUNT_ADDR)
|
|
|
|
#endif /* defined(CONFIG_MPC8260) */
|
|
|
|
|
2010-02-18 07:08:25 +00:00
|
|
|
#if defined(CONFIG_QE)
|
2014-06-03 08:27:07 +00:00
|
|
|
#include <linux/immap_qe.h>
|
2010-04-28 08:47:36 +00:00
|
|
|
#define CONFIG_SYS_BOOTCOUNT_ADDR (CONFIG_SYS_IMMR + 0x110000 + \
|
|
|
|
QE_MURAM_SIZE - 2 * sizeof(u32))
|
2015-12-22 12:15:14 +00:00
|
|
|
#endif /* defined(CONFIG_QE) */
|
2010-04-28 08:47:36 +00:00
|
|
|
|
|
|
|
#if defined(CONFIG_4xx)
|
|
|
|
#define CONFIG_SYS_BOOTCOUNT_ADDR (CONFIG_SYS_OCM_DATA_ADDR + \
|
|
|
|
CONFIG_SYS_BOOTCOUNT_ADDR)
|
|
|
|
#endif /* defined(CONFIG_4xx) */
|
|
|
|
|
|
|
|
#endif /* !defined(CONFIG_SYS_BOOTCOUNT_ADDR) */
|
|
|
|
|
2012-08-16 17:55:41 +00:00
|
|
|
/* Now implement the generic default functions */
|
|
|
|
#if defined(CONFIG_SYS_BOOTCOUNT_ADDR)
|
|
|
|
__weak void bootcount_store(ulong a)
|
2010-04-28 08:47:36 +00:00
|
|
|
{
|
|
|
|
void *reg = (void *)CONFIG_SYS_BOOTCOUNT_ADDR;
|
|
|
|
|
|
|
|
#if defined(CONFIG_SYS_BOOTCOUNT_SINGLEWORD)
|
2012-08-16 17:55:41 +00:00
|
|
|
raw_bootcount_store(reg, (BOOTCOUNT_MAGIC & 0xffff0000) | a);
|
2010-04-28 08:47:36 +00:00
|
|
|
#else
|
2012-08-16 17:55:41 +00:00
|
|
|
raw_bootcount_store(reg, a);
|
|
|
|
raw_bootcount_store(reg + 4, BOOTCOUNT_MAGIC);
|
2015-12-22 12:15:14 +00:00
|
|
|
#endif /* defined(CONFIG_SYS_BOOTCOUNT_SINGLEWORD */
|
2010-04-28 08:47:36 +00:00
|
|
|
}
|
|
|
|
|
2012-08-16 17:55:41 +00:00
|
|
|
__weak ulong bootcount_load(void)
|
2010-04-28 08:47:36 +00:00
|
|
|
{
|
|
|
|
void *reg = (void *)CONFIG_SYS_BOOTCOUNT_ADDR;
|
|
|
|
|
|
|
|
#if defined(CONFIG_SYS_BOOTCOUNT_SINGLEWORD)
|
2012-08-16 17:55:41 +00:00
|
|
|
u32 tmp = raw_bootcount_load(reg);
|
2010-05-20 14:09:35 +00:00
|
|
|
|
|
|
|
if ((tmp & 0xffff0000) != (BOOTCOUNT_MAGIC & 0xffff0000))
|
2010-04-28 08:47:36 +00:00
|
|
|
return 0;
|
|
|
|
else
|
2010-05-20 14:09:35 +00:00
|
|
|
return (tmp & 0x0000ffff);
|
2010-04-28 08:47:36 +00:00
|
|
|
#else
|
2012-08-16 17:55:41 +00:00
|
|
|
if (raw_bootcount_load(reg + 4) != BOOTCOUNT_MAGIC)
|
2010-04-28 08:47:36 +00:00
|
|
|
return 0;
|
|
|
|
else
|
2012-08-16 17:55:41 +00:00
|
|
|
return raw_bootcount_load(reg);
|
2015-12-22 12:15:14 +00:00
|
|
|
#endif /* defined(CONFIG_SYS_BOOTCOUNT_SINGLEWORD) */
|
2010-04-28 08:47:36 +00:00
|
|
|
}
|
2015-12-22 12:15:14 +00:00
|
|
|
#endif /* defined(CONFIG_SYS_BOOTCOUNT_ADDR) */
|