mirror of
				https://xff.cz/git/u-boot/
				synced 2025-10-26 16:13:55 +01:00 
			
		
		
		
	The original bootcount methods do not provide an interface to DM and rely on a static configuration for I2C devices (e.g. bus, chip-addr, etc. are configured through defines statically). On a modern system that exposes multiple devices in a DTS-configurable way, this is less than optimal and a interface to DM-based devices will be desirable. This adds a simple driver that is DM-aware and configurable via DTS. If ambiguous (i.e. multiple bootcount-devices are present) the /chosen/u-boot,bootcount-device property can be used to select one bootcount device. Initially, this provides support for the following DM devices: * RTC devices Signed-off-by: Philipp Tomsich <philipp.tomsich@theobroma-systems.com> Tested-by: Klaus Goger <klaus.goger@theobroma-systems.com>
		
			
				
	
	
		
			138 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			138 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* SPDX-License-Identifier: GPL-2.0+ */
 | |
| /*
 | |
|  * (C) Copyright 2012
 | |
|  * Stefan Roese, DENX Software Engineering, sr@denx.de.
 | |
|  */
 | |
| #ifndef _BOOTCOUNT_H__
 | |
| #define _BOOTCOUNT_H__
 | |
| 
 | |
| #include <common.h>
 | |
| #include <asm/io.h>
 | |
| #include <asm/byteorder.h>
 | |
| 
 | |
| #ifdef CONFIG_DM_BOOTCOUNT
 | |
| 
 | |
| struct bootcount_ops {
 | |
| 	/**
 | |
| 	 * get() - get the current bootcount value
 | |
| 	 *
 | |
| 	 * Returns the current counter value of the bootcount backing
 | |
| 	 * store.
 | |
| 	 *
 | |
| 	 * @dev:	Device to read from
 | |
| 	 * @bootcount:	Address to put the current bootcount value
 | |
| 	 */
 | |
| 	int (*get)(struct udevice *dev, u32 *bootcount);
 | |
| 
 | |
| 	/**
 | |
| 	 * set() - set a bootcount value (e.g. to reset or increment)
 | |
| 	 *
 | |
| 	 * Sets the value in the bootcount backing store.
 | |
| 	 *
 | |
| 	 * @dev:	Device to read from
 | |
| 	 * @bootcount:	New bootcount value to store
 | |
| 	 */
 | |
| 	int (*set)(struct udevice *dev, const u32 bootcount);
 | |
| };
 | |
| 
 | |
| /* Access the operations for a bootcount device */
 | |
| #define bootcount_get_ops(dev)	((struct bootcount_ops *)(dev)->driver->ops)
 | |
| 
 | |
| /**
 | |
|  * dm_bootcount_get() - Read the current value from a bootcount storage
 | |
|  *
 | |
|  * @dev:	Device to read from
 | |
|  * @bootcount:	Place to put the current bootcount
 | |
|  * @return 0 if OK, -ve on error
 | |
|  */
 | |
| int dm_bootcount_get(struct udevice *dev, u32 *bootcount);
 | |
| 
 | |
| /**
 | |
|  * dm_bootcount_set() - Write a value to a bootcount storage
 | |
|  *
 | |
|  * @dev:	Device to read from
 | |
|  * @bootcount:  Value to be written to the backing storage
 | |
|  * @return 0 if OK, -ve on error
 | |
|  */
 | |
| int dm_bootcount_set(struct udevice *dev, u32 bootcount);
 | |
| 
 | |
| #endif
 | |
| 
 | |
| #if defined(CONFIG_SPL_BOOTCOUNT_LIMIT) || defined(CONFIG_BOOTCOUNT_LIMIT)
 | |
| 
 | |
| #if !defined(CONFIG_SYS_BOOTCOUNT_LE) && !defined(CONFIG_SYS_BOOTCOUNT_BE)
 | |
| # if __BYTE_ORDER == __LITTLE_ENDIAN
 | |
| #  define CONFIG_SYS_BOOTCOUNT_LE
 | |
| # else
 | |
| #  define CONFIG_SYS_BOOTCOUNT_BE
 | |
| # endif
 | |
| #endif
 | |
| 
 | |
| #ifdef CONFIG_SYS_BOOTCOUNT_LE
 | |
| static inline void raw_bootcount_store(volatile u32 *addr, u32 data)
 | |
| {
 | |
| 	out_le32(addr, data);
 | |
| }
 | |
| 
 | |
| static inline u32 raw_bootcount_load(volatile u32 *addr)
 | |
| {
 | |
| 	return in_le32(addr);
 | |
| }
 | |
| #else
 | |
| static inline void raw_bootcount_store(volatile u32 *addr, u32 data)
 | |
| {
 | |
| 	out_be32(addr, data);
 | |
| }
 | |
| 
 | |
| static inline u32 raw_bootcount_load(volatile u32 *addr)
 | |
| {
 | |
| 	return in_be32(addr);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| DECLARE_GLOBAL_DATA_PTR;
 | |
| static inline int bootcount_error(void)
 | |
| {
 | |
| 	unsigned long bootcount = bootcount_load();
 | |
| 	unsigned long bootlimit = env_get_ulong("bootlimit", 10, 0);
 | |
| 
 | |
| 	if (bootlimit && bootcount > bootlimit) {
 | |
| 		printf("Warning: Bootlimit (%lu) exceeded.", bootlimit);
 | |
| 		if (!(gd->flags & GD_FLG_SPL_INIT))
 | |
| 			printf(" Using altbootcmd.");
 | |
| 		printf("\n");
 | |
| 
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static inline void bootcount_inc(void)
 | |
| {
 | |
| 	unsigned long bootcount = bootcount_load();
 | |
| 
 | |
| 	if (gd->flags & GD_FLG_SPL_INIT) {
 | |
| 		bootcount_store(++bootcount);
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| #ifndef CONFIG_SPL_BUILD
 | |
| 	/* Only increment bootcount when no bootcount support in SPL */
 | |
| #ifndef CONFIG_SPL_BOOTCOUNT_LIMIT
 | |
| 	bootcount_store(++bootcount);
 | |
| #endif
 | |
| 	env_set_ulong("bootcount", bootcount);
 | |
| #endif /* !CONFIG_SPL_BUILD */
 | |
| }
 | |
| 
 | |
| #if defined(CONFIG_SPL_BUILD) && !defined(CONFIG_SPL_BOOTCOUNT_LIMIT)
 | |
| void bootcount_store(ulong a) {};
 | |
| ulong bootcount_load(void) { return 0; }
 | |
| #endif /* CONFIG_SPL_BUILD && !CONFIG_SPL_BOOTCOUNT_LIMIT */
 | |
| #else
 | |
| static inline int bootcount_error(void) { return 0; }
 | |
| static inline void bootcount_inc(void) {}
 | |
| #endif /* CONFIG_SPL_BOOTCOUNT_LIMIT || CONFIG_BOOTCOUNT_LIMIT */
 | |
| #endif /* _BOOTCOUNT_H__ */
 |