blob: 6fda6efb1cea1533a62649e63746b992df249cbd [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Michal Simeka0d28022013-11-21 13:39:02 -08002/*
3 * Copyright (C) 2013 Xilinx, Inc.
Michal Simeka0d28022013-11-21 13:39:02 -08004 */
Michal Simeka0d28022013-11-21 13:39:02 -08005#include <command.h>
6#include <clk.h>
Marek Vasut9254f992018-08-08 22:10:44 +02007#if defined(CONFIG_DM) && defined(CONFIG_CLK)
8#include <dm.h>
Peng Fan51709f72019-08-21 13:35:14 +00009#include <dm/device.h>
10#include <dm/root.h>
Marek Vasut9254f992018-08-08 22:10:44 +020011#include <dm/device-internal.h>
Peng Fan51709f72019-08-21 13:35:14 +000012#include <linux/clk-provider.h>
Marek Vasut9254f992018-08-08 22:10:44 +020013#endif
Michal Simeka0d28022013-11-21 13:39:02 -080014
Marek Vasut9254f992018-08-08 22:10:44 +020015#if defined(CONFIG_DM) && defined(CONFIG_CLK)
Peng Fan51709f72019-08-21 13:35:14 +000016static void show_clks(struct udevice *dev, int depth, int last_flag)
17{
18 int i, is_last;
19 struct udevice *child;
Tero Kristo1a22eba2021-06-11 11:45:07 +030020 struct clk *clkp, *parent;
Peng Fan51709f72019-08-21 13:35:14 +000021 u32 rate;
Marek Vasut9254f992018-08-08 22:10:44 +020022
Peng Fan51709f72019-08-21 13:35:14 +000023 clkp = dev_get_clk_ptr(dev);
Patrick Delaunayaac5b9a2022-12-13 14:57:10 +010024 if (clkp) {
Tero Kristo1a22eba2021-06-11 11:45:07 +030025 parent = clk_get_parent(clkp);
26 if (!IS_ERR(parent) && depth == -1)
27 return;
Patrick Delaunay0ca4cc02020-07-30 14:04:10 +020028 depth++;
Peng Fan51709f72019-08-21 13:35:14 +000029 rate = clk_get_rate(clkp);
Marek Vasut9254f992018-08-08 22:10:44 +020030
Patrick Delaunay15c5a342020-07-30 14:04:09 +020031 printf(" %-12u %8d ", rate, clkp->enable_count);
Marek Vasut9254f992018-08-08 22:10:44 +020032
Patrick Delaunay15c5a342020-07-30 14:04:09 +020033 for (i = depth; i >= 0; i--) {
34 is_last = (last_flag >> i) & 1;
35 if (i) {
36 if (is_last)
37 printf(" ");
38 else
39 printf("| ");
40 } else {
41 if (is_last)
42 printf("`-- ");
43 else
44 printf("|-- ");
45 }
Peng Fan51709f72019-08-21 13:35:14 +000046 }
Marek Vasut9254f992018-08-08 22:10:44 +020047
Patrick Delaunay15c5a342020-07-30 14:04:09 +020048 printf("%s\n", dev->name);
Peng Fan51709f72019-08-21 13:35:14 +000049 }
Ismael Luceno Cortes8b5d3a42019-03-18 12:27:32 +000050
Patrick Delaunayaac5b9a2022-12-13 14:57:10 +010051 device_foreach_child_probe(child, dev) {
52 if (device_get_uclass_id(child) != UCLASS_CLK)
53 continue;
Tero Kristo1a22eba2021-06-11 11:45:07 +030054 if (child == dev)
55 continue;
Peng Fan51709f72019-08-21 13:35:14 +000056 is_last = list_is_last(&child->sibling_node, &dev->child_head);
Patrick Delaunay0ca4cc02020-07-30 14:04:10 +020057 show_clks(child, depth, (last_flag << 1) | is_last);
Peng Fan51709f72019-08-21 13:35:14 +000058 }
59}
Ismael Luceno Cortes8b5d3a42019-03-18 12:27:32 +000060
Igor Prusov7336e682023-11-09 13:55:16 +030061static int soc_clk_dump(void)
Peng Fan51709f72019-08-21 13:35:14 +000062{
Tero Kristo1a22eba2021-06-11 11:45:07 +030063 struct udevice *dev;
Igor Prusov6b565402023-11-09 13:55:14 +030064 const struct clk_ops *ops;
Tero Kristo1a22eba2021-06-11 11:45:07 +030065
66 printf(" Rate Usecnt Name\n");
67 printf("------------------------------------------\n");
68
Patrick Delaunayaac5b9a2022-12-13 14:57:10 +010069 uclass_foreach_dev_probe(UCLASS_CLK, dev)
Tero Kristo1a22eba2021-06-11 11:45:07 +030070 show_clks(dev, -1, 0);
Marek Vasut9254f992018-08-08 22:10:44 +020071
Igor Prusov6b565402023-11-09 13:55:14 +030072 uclass_foreach_dev_probe(UCLASS_CLK, dev) {
73 ops = dev_get_driver_ops(dev);
74 if (ops && ops->dump) {
75 printf("\n%s %s:\n", dev->driver->name, dev->name);
76 ops->dump(dev);
77 }
78 }
79
Marek Vasut9254f992018-08-08 22:10:44 +020080 return 0;
Peng Fan51709f72019-08-21 13:35:14 +000081}
Marek Vasut9254f992018-08-08 22:10:44 +020082#else
Igor Prusov7336e682023-11-09 13:55:16 +030083static int soc_clk_dump(void)
Peng Fan51709f72019-08-21 13:35:14 +000084{
Michal Simeka0d28022013-11-21 13:39:02 -080085 puts("Not implemented\n");
86 return 1;
87}
Peng Fan51709f72019-08-21 13:35:14 +000088#endif
Michal Simeka0d28022013-11-21 13:39:02 -080089
Simon Glassed38aef2020-05-10 11:40:03 -060090static int do_clk_dump(struct cmd_tbl *cmdtp, int flag, int argc,
Michal Simeka0d28022013-11-21 13:39:02 -080091 char *const argv[])
92{
Michal Simek7f6992b2018-04-19 15:15:25 +020093 int ret;
94
95 ret = soc_clk_dump();
96 if (ret < 0) {
97 printf("Clock dump error %d\n", ret);
98 ret = CMD_RET_FAILURE;
99 }
100
101 return ret;
Michal Simeka0d28022013-11-21 13:39:02 -0800102}
103
Tero Kristo4bd4cb42021-06-11 11:45:09 +0300104#if CONFIG_IS_ENABLED(DM) && CONFIG_IS_ENABLED(CLK)
Tero Kristo4bd4cb42021-06-11 11:45:09 +0300105static int do_clk_setfreq(struct cmd_tbl *cmdtp, int flag, int argc,
106 char *const argv[])
107{
108 struct clk *clk = NULL;
109 s32 freq;
110 struct udevice *dev;
111
Patrick Delaunay5c6d32d2022-01-31 17:21:37 +0100112 if (argc != 3)
113 return CMD_RET_USAGE;
114
Simon Glassff9b9032021-07-24 09:03:30 -0600115 freq = dectoul(argv[2], NULL);
Tero Kristo4bd4cb42021-06-11 11:45:09 +0300116
Patrick Delaunay27e14d42022-01-31 17:21:38 +0100117 if (!uclass_get_device_by_name(UCLASS_CLK, argv[1], &dev))
Tero Kristo4bd4cb42021-06-11 11:45:09 +0300118 clk = dev_get_clk_ptr(dev);
119
120 if (!clk) {
121 printf("clock '%s' not found.\n", argv[1]);
Patrick Delaunayfe23c802022-01-31 17:21:39 +0100122 return CMD_RET_FAILURE;
Tero Kristo4bd4cb42021-06-11 11:45:09 +0300123 }
124
125 freq = clk_set_rate(clk, freq);
126 if (freq < 0) {
127 printf("set_rate failed: %d\n", freq);
128 return CMD_RET_FAILURE;
129 }
130
131 printf("set_rate returns %u\n", freq);
132 return 0;
133}
134#endif
135
Simon Glassed38aef2020-05-10 11:40:03 -0600136static struct cmd_tbl cmd_clk_sub[] = {
Michal Simeka0d28022013-11-21 13:39:02 -0800137 U_BOOT_CMD_MKENT(dump, 1, 1, do_clk_dump, "", ""),
Tero Kristo4bd4cb42021-06-11 11:45:09 +0300138#if CONFIG_IS_ENABLED(DM) && CONFIG_IS_ENABLED(CLK)
139 U_BOOT_CMD_MKENT(setfreq, 3, 1, do_clk_setfreq, "", ""),
140#endif
Michal Simeka0d28022013-11-21 13:39:02 -0800141};
142
Simon Glassed38aef2020-05-10 11:40:03 -0600143static int do_clk(struct cmd_tbl *cmdtp, int flag, int argc,
Michal Simeka0d28022013-11-21 13:39:02 -0800144 char *const argv[])
145{
Simon Glassed38aef2020-05-10 11:40:03 -0600146 struct cmd_tbl *c;
Michal Simeka0d28022013-11-21 13:39:02 -0800147
148 if (argc < 2)
149 return CMD_RET_USAGE;
150
151 /* Strip off leading 'clk' command argument */
152 argc--;
153 argv++;
154
155 c = find_cmd_tbl(argv[0], &cmd_clk_sub[0], ARRAY_SIZE(cmd_clk_sub));
156
157 if (c)
158 return c->cmd(cmdtp, flag, argc, argv);
159 else
160 return CMD_RET_USAGE;
161}
162
Tom Rini03f146c2023-10-07 15:13:08 -0400163U_BOOT_LONGHELP(clk,
Tero Kristo4bd4cb42021-06-11 11:45:09 +0300164 "dump - Print clock frequencies\n"
Tom Rini03f146c2023-10-07 15:13:08 -0400165 "clk setfreq [clk] [freq] - Set clock frequency");
Michal Simeka0d28022013-11-21 13:39:02 -0800166
Tero Kristo4bd4cb42021-06-11 11:45:09 +0300167U_BOOT_CMD(clk, 4, 1, do_clk, "CLK sub-system", clk_help_text);