blob: 491f2140a32c0ac1d1296501a732a776eef73834 [file] [log] [blame]
Tom Rini83d290c2018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Michal Simek08d0d6f2013-11-21 13:39:02 -08002/*
3 * Copyright (C) 2013 Xilinx, Inc.
Michal Simek08d0d6f2013-11-21 13:39:02 -08004 */
5#include <common.h>
6#include <command.h>
7#include <clk.h>
Marek Vasutff8eee02018-08-08 22:10:44 +02008#if defined(CONFIG_DM) && defined(CONFIG_CLK)
9#include <dm.h>
Peng Fanaeeb2e62019-08-21 13:35:14 +000010#include <dm/device.h>
11#include <dm/root.h>
Marek Vasutff8eee02018-08-08 22:10:44 +020012#include <dm/device-internal.h>
Peng Fanaeeb2e62019-08-21 13:35:14 +000013#include <linux/clk-provider.h>
Marek Vasutff8eee02018-08-08 22:10:44 +020014#endif
Michal Simek08d0d6f2013-11-21 13:39:02 -080015
Peng Fanaeeb2e62019-08-21 13:35:14 +000016#if defined(CONFIG_DM) && defined(CONFIG_CLK)
17static void show_clks(struct udevice *dev, int depth, int last_flag)
18{
19 int i, is_last;
20 struct udevice *child;
Tero Kristo1a725e22021-06-11 11:45:07 +030021 struct clk *clkp, *parent;
Peng Fanaeeb2e62019-08-21 13:35:14 +000022 u32 rate;
23
Caleb Connollyd3888992023-10-02 16:56:21 +010024 clk_dump_clks(dev);
25
Peng Fanaeeb2e62019-08-21 13:35:14 +000026 clkp = dev_get_clk_ptr(dev);
Patrick Delaunayc40251c2022-12-13 14:57:10 +010027 if (clkp) {
Tero Kristo1a725e22021-06-11 11:45:07 +030028 parent = clk_get_parent(clkp);
29 if (!IS_ERR(parent) && depth == -1)
30 return;
Patrick Delaunay689ca8c2020-07-30 14:04:10 +020031 depth++;
Peng Fanaeeb2e62019-08-21 13:35:14 +000032 rate = clk_get_rate(clkp);
33
Patrick Delaunaycc632842020-07-30 14:04:09 +020034 printf(" %-12u %8d ", rate, clkp->enable_count);
Peng Fanaeeb2e62019-08-21 13:35:14 +000035
Patrick Delaunaycc632842020-07-30 14:04:09 +020036 for (i = depth; i >= 0; i--) {
37 is_last = (last_flag >> i) & 1;
38 if (i) {
39 if (is_last)
40 printf(" ");
41 else
42 printf("| ");
43 } else {
44 if (is_last)
45 printf("`-- ");
46 else
47 printf("|-- ");
48 }
Peng Fanaeeb2e62019-08-21 13:35:14 +000049 }
Peng Fanaeeb2e62019-08-21 13:35:14 +000050
Patrick Delaunaycc632842020-07-30 14:04:09 +020051 printf("%s\n", dev->name);
Peng Fanaeeb2e62019-08-21 13:35:14 +000052 }
53
Patrick Delaunayc40251c2022-12-13 14:57:10 +010054 device_foreach_child_probe(child, dev) {
55 if (device_get_uclass_id(child) != UCLASS_CLK)
56 continue;
Tero Kristo1a725e22021-06-11 11:45:07 +030057 if (child == dev)
58 continue;
Peng Fanaeeb2e62019-08-21 13:35:14 +000059 is_last = list_is_last(&child->sibling_node, &dev->child_head);
Patrick Delaunay689ca8c2020-07-30 14:04:10 +020060 show_clks(child, depth, (last_flag << 1) | is_last);
Peng Fanaeeb2e62019-08-21 13:35:14 +000061 }
62}
63
Michal Simek08d0d6f2013-11-21 13:39:02 -080064int __weak soc_clk_dump(void)
65{
Tero Kristo1a725e22021-06-11 11:45:07 +030066 struct udevice *dev;
Tero Kristo1a725e22021-06-11 11:45:07 +030067
68 printf(" Rate Usecnt Name\n");
69 printf("------------------------------------------\n");
70
Patrick Delaunayc40251c2022-12-13 14:57:10 +010071 uclass_foreach_dev_probe(UCLASS_CLK, dev)
Tero Kristo1a725e22021-06-11 11:45:07 +030072 show_clks(dev, -1, 0);
Marek Vasutff8eee02018-08-08 22:10:44 +020073
74 return 0;
Peng Fanaeeb2e62019-08-21 13:35:14 +000075}
Marek Vasutff8eee02018-08-08 22:10:44 +020076#else
Peng Fanaeeb2e62019-08-21 13:35:14 +000077int __weak soc_clk_dump(void)
78{
Michal Simek08d0d6f2013-11-21 13:39:02 -080079 puts("Not implemented\n");
80 return 1;
81}
Peng Fanaeeb2e62019-08-21 13:35:14 +000082#endif
Michal Simek08d0d6f2013-11-21 13:39:02 -080083
Simon Glass09140112020-05-10 11:40:03 -060084static int do_clk_dump(struct cmd_tbl *cmdtp, int flag, int argc,
Michal Simek08d0d6f2013-11-21 13:39:02 -080085 char *const argv[])
86{
Michal Simekebc675b2018-04-19 15:15:25 +020087 int ret;
88
89 ret = soc_clk_dump();
90 if (ret < 0) {
91 printf("Clock dump error %d\n", ret);
92 ret = CMD_RET_FAILURE;
93 }
94
95 return ret;
Michal Simek08d0d6f2013-11-21 13:39:02 -080096}
97
Tero Kristo7ab418f2021-06-11 11:45:09 +030098#if CONFIG_IS_ENABLED(DM) && CONFIG_IS_ENABLED(CLK)
Tero Kristo7ab418f2021-06-11 11:45:09 +030099static int do_clk_setfreq(struct cmd_tbl *cmdtp, int flag, int argc,
100 char *const argv[])
101{
102 struct clk *clk = NULL;
103 s32 freq;
104 struct udevice *dev;
105
Patrick Delaunay3386fb12022-01-31 17:21:37 +0100106 if (argc != 3)
107 return CMD_RET_USAGE;
108
Simon Glass0b1284e2021-07-24 09:03:30 -0600109 freq = dectoul(argv[2], NULL);
Tero Kristo7ab418f2021-06-11 11:45:09 +0300110
Patrick Delaunayafcc2612022-01-31 17:21:38 +0100111 if (!uclass_get_device_by_name(UCLASS_CLK, argv[1], &dev))
Tero Kristo7ab418f2021-06-11 11:45:09 +0300112 clk = dev_get_clk_ptr(dev);
113
114 if (!clk) {
115 printf("clock '%s' not found.\n", argv[1]);
Patrick Delaunay534859a2022-01-31 17:21:39 +0100116 return CMD_RET_FAILURE;
Tero Kristo7ab418f2021-06-11 11:45:09 +0300117 }
118
119 freq = clk_set_rate(clk, freq);
120 if (freq < 0) {
121 printf("set_rate failed: %d\n", freq);
122 return CMD_RET_FAILURE;
123 }
124
125 printf("set_rate returns %u\n", freq);
126 return 0;
127}
128#endif
129
Simon Glass09140112020-05-10 11:40:03 -0600130static struct cmd_tbl cmd_clk_sub[] = {
Michal Simek08d0d6f2013-11-21 13:39:02 -0800131 U_BOOT_CMD_MKENT(dump, 1, 1, do_clk_dump, "", ""),
Tero Kristo7ab418f2021-06-11 11:45:09 +0300132#if CONFIG_IS_ENABLED(DM) && CONFIG_IS_ENABLED(CLK)
133 U_BOOT_CMD_MKENT(setfreq, 3, 1, do_clk_setfreq, "", ""),
134#endif
Michal Simek08d0d6f2013-11-21 13:39:02 -0800135};
136
Simon Glass09140112020-05-10 11:40:03 -0600137static int do_clk(struct cmd_tbl *cmdtp, int flag, int argc,
Michal Simek08d0d6f2013-11-21 13:39:02 -0800138 char *const argv[])
139{
Simon Glass09140112020-05-10 11:40:03 -0600140 struct cmd_tbl *c;
Michal Simek08d0d6f2013-11-21 13:39:02 -0800141
142 if (argc < 2)
143 return CMD_RET_USAGE;
144
145 /* Strip off leading 'clk' command argument */
146 argc--;
147 argv++;
148
149 c = find_cmd_tbl(argv[0], &cmd_clk_sub[0], ARRAY_SIZE(cmd_clk_sub));
150
151 if (c)
152 return c->cmd(cmdtp, flag, argc, argv);
153 else
154 return CMD_RET_USAGE;
155}
156
Tom Rini36162182023-10-07 15:13:08 -0400157U_BOOT_LONGHELP(clk,
Tero Kristo7ab418f2021-06-11 11:45:09 +0300158 "dump - Print clock frequencies\n"
Tom Rini36162182023-10-07 15:13:08 -0400159 "clk setfreq [clk] [freq] - Set clock frequency");
Michal Simek08d0d6f2013-11-21 13:39:02 -0800160
Tero Kristo7ab418f2021-06-11 11:45:09 +0300161U_BOOT_CMD(clk, 4, 1, do_clk, "CLK sub-system", clk_help_text);