mirror of
				https://xff.cz/git/u-boot/
				synced 2025-10-31 10:26:10 +01:00 
			
		
		
		
	Clocks are an important feature of platforms and have become increasing complex with time. Most modern SoCs have multiple PLLs and dozens of clock dividers which distribute clocks to on-chip peripherals. Some SoC implementations have a clock API which is private to that SoC family, e.g. Tegra and Exynos. This is useful but it would be better to have a common API that can be understood and used throughout U-Boot. Add a simple clock API as a starting point. It supports querying and setting the rate of a clock. Each clock is a device. To reduce memory and processing overhead the concept of peripheral clocks is provided. These do not need to be explicit devices - it is possible to write a driver that can adjust the I2C clock (for example) without an explicit I2C clock device. This can dramatically reduce the number of devices (and associated overhead) in a complex SoC. Clocks are referenced by a number, and it is expected that SoCs will define that numbering themselves via an enum. Signed-off-by: Simon Glass <sjg@chromium.org>
		
			
				
	
	
		
			59 lines
		
	
	
		
			1.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			59 lines
		
	
	
		
			1.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Copyright (C) 2015 Google, Inc
 | |
|  * Written by Simon Glass <sjg@chromium.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier:	GPL-2.0+
 | |
|  */
 | |
| 
 | |
| #include <common.h>
 | |
| #include <clk.h>
 | |
| #include <dm.h>
 | |
| #include <errno.h>
 | |
| #include <dm/lists.h>
 | |
| #include <dm/root.h>
 | |
| 
 | |
| ulong clk_get_rate(struct udevice *dev)
 | |
| {
 | |
| 	struct clk_ops *ops = clk_get_ops(dev);
 | |
| 
 | |
| 	if (!ops->get_rate)
 | |
| 		return -ENOSYS;
 | |
| 
 | |
| 	return ops->get_rate(dev);
 | |
| }
 | |
| 
 | |
| ulong clk_set_rate(struct udevice *dev, ulong rate)
 | |
| {
 | |
| 	struct clk_ops *ops = clk_get_ops(dev);
 | |
| 
 | |
| 	if (!ops->set_rate)
 | |
| 		return -ENOSYS;
 | |
| 
 | |
| 	return ops->set_rate(dev, rate);
 | |
| }
 | |
| 
 | |
| ulong clk_get_periph_rate(struct udevice *dev, int periph)
 | |
| {
 | |
| 	struct clk_ops *ops = clk_get_ops(dev);
 | |
| 
 | |
| 	if (!ops->get_periph_rate)
 | |
| 		return -ENOSYS;
 | |
| 
 | |
| 	return ops->get_periph_rate(dev, periph);
 | |
| }
 | |
| 
 | |
| ulong clk_set_periph_rate(struct udevice *dev, int periph, ulong rate)
 | |
| {
 | |
| 	struct clk_ops *ops = clk_get_ops(dev);
 | |
| 
 | |
| 	if (!ops->set_periph_rate)
 | |
| 		return -ENOSYS;
 | |
| 
 | |
| 	return ops->set_periph_rate(dev, periph, rate);
 | |
| }
 | |
| 
 | |
| UCLASS_DRIVER(clk) = {
 | |
| 	.id		= UCLASS_CLK,
 | |
| 	.name		= "clk",
 | |
| };
 |