106811959SSimon Glass /* 206811959SSimon Glass * Copyright (c) 2013 Google, Inc 306811959SSimon Glass * 406811959SSimon Glass * (C) Copyright 2012 506811959SSimon Glass * Marek Vasut <marex@denx.de> 606811959SSimon Glass * 706811959SSimon Glass * SPDX-License-Identifier: GPL-2.0+ 806811959SSimon Glass */ 906811959SSimon Glass 1006811959SSimon Glass #include <common.h> 1106811959SSimon Glass #include <dm.h> 1206811959SSimon Glass #include <malloc.h> 1306811959SSimon Glass #include <errno.h> 1406811959SSimon Glass #include <asm/io.h> 1506811959SSimon Glass #include <dm/root.h> 1606811959SSimon Glass #include <dm/test.h> 1706811959SSimon Glass #include <dm/uclass-internal.h> 1806811959SSimon Glass 19*a56642c7SMasahiro Yamada static void show_devices(struct udevice *dev, int depth, int last_flag) 20fffa24d7SSimon Glass { 21*a56642c7SMasahiro Yamada int i, is_last; 22*a56642c7SMasahiro Yamada struct udevice *child; 23*a56642c7SMasahiro Yamada char class_name[12]; 24*a56642c7SMasahiro Yamada 25*a56642c7SMasahiro Yamada /* print the first 11 characters to not break the tree-format. */ 26*a56642c7SMasahiro Yamada strlcpy(class_name, dev->uclass->uc_drv->name, sizeof(class_name)); 27*a56642c7SMasahiro Yamada printf(" %-11s [ %c ] ", class_name, 28*a56642c7SMasahiro Yamada dev->flags & DM_FLAG_ACTIVATED ? '+' : ' '); 29*a56642c7SMasahiro Yamada 30*a56642c7SMasahiro Yamada for (i = depth; i >= 0; i--) { 31*a56642c7SMasahiro Yamada is_last = (last_flag >> i) & 1; 32*a56642c7SMasahiro Yamada if (i) { 33*a56642c7SMasahiro Yamada if (is_last) 34*a56642c7SMasahiro Yamada printf(" "); 35*a56642c7SMasahiro Yamada else 36*a56642c7SMasahiro Yamada printf("| "); 37*a56642c7SMasahiro Yamada } else { 38*a56642c7SMasahiro Yamada if (is_last) 39*a56642c7SMasahiro Yamada printf("`-- "); 40*a56642c7SMasahiro Yamada else 41*a56642c7SMasahiro Yamada printf("|-- "); 42*a56642c7SMasahiro Yamada } 43fffa24d7SSimon Glass } 44fffa24d7SSimon Glass 45*a56642c7SMasahiro Yamada printf("%s\n", dev->name); 4606811959SSimon Glass 47*a56642c7SMasahiro Yamada list_for_each_entry(child, &dev->child_head, sibling_node) { 48*a56642c7SMasahiro Yamada is_last = list_is_last(&child->sibling_node, &dev->child_head); 49*a56642c7SMasahiro Yamada show_devices(child, depth + 1, (last_flag << 1) | is_last); 5006811959SSimon Glass } 5106811959SSimon Glass } 5206811959SSimon Glass 5306811959SSimon Glass static int do_dm_dump_all(cmd_tbl_t *cmdtp, int flag, int argc, 5406811959SSimon Glass char * const argv[]) 5506811959SSimon Glass { 5654c5d08aSHeiko Schocher struct udevice *root; 5706811959SSimon Glass 5806811959SSimon Glass root = dm_root(); 59*a56642c7SMasahiro Yamada if (root) { 60*a56642c7SMasahiro Yamada printf(" Class Probed Name\n"); 61*a56642c7SMasahiro Yamada printf("----------------------------------------\n"); 62*a56642c7SMasahiro Yamada show_devices(root, -1, 0); 63*a56642c7SMasahiro Yamada } 64*a56642c7SMasahiro Yamada 65*a56642c7SMasahiro Yamada return 0; 66*a56642c7SMasahiro Yamada } 67*a56642c7SMasahiro Yamada 68*a56642c7SMasahiro Yamada /** 69*a56642c7SMasahiro Yamada * dm_display_line() - Display information about a single device 70*a56642c7SMasahiro Yamada * 71*a56642c7SMasahiro Yamada * Displays a single line of information with an option prefix 72*a56642c7SMasahiro Yamada * 73*a56642c7SMasahiro Yamada * @dev: Device to display 74*a56642c7SMasahiro Yamada */ 75*a56642c7SMasahiro Yamada static void dm_display_line(struct udevice *dev) 76*a56642c7SMasahiro Yamada { 77*a56642c7SMasahiro Yamada printf("- %c %s @ %08lx", 78*a56642c7SMasahiro Yamada dev->flags & DM_FLAG_ACTIVATED ? '*' : ' ', 79*a56642c7SMasahiro Yamada dev->name, (ulong)map_to_sysmem(dev)); 80*a56642c7SMasahiro Yamada if (dev->req_seq != -1) 81*a56642c7SMasahiro Yamada printf(", %d", dev->req_seq); 82*a56642c7SMasahiro Yamada puts("\n"); 8306811959SSimon Glass } 8406811959SSimon Glass 8506811959SSimon Glass static int do_dm_dump_uclass(cmd_tbl_t *cmdtp, int flag, int argc, 8606811959SSimon Glass char * const argv[]) 8706811959SSimon Glass { 8806811959SSimon Glass struct uclass *uc; 8906811959SSimon Glass int ret; 9006811959SSimon Glass int id; 9106811959SSimon Glass 9206811959SSimon Glass for (id = 0; id < UCLASS_COUNT; id++) { 9354c5d08aSHeiko Schocher struct udevice *dev; 9406811959SSimon Glass 9506811959SSimon Glass ret = uclass_get(id, &uc); 9606811959SSimon Glass if (ret) 9706811959SSimon Glass continue; 9806811959SSimon Glass 9906811959SSimon Glass printf("uclass %d: %s\n", id, uc->uc_drv->name); 1004e8bc211SSimon Glass if (list_empty(&uc->dev_head)) 1014e8bc211SSimon Glass continue; 1024e8bc211SSimon Glass list_for_each_entry(dev, &uc->dev_head, uclass_node) { 103*a56642c7SMasahiro Yamada dm_display_line(dev); 10406811959SSimon Glass } 10506811959SSimon Glass puts("\n"); 10606811959SSimon Glass } 10706811959SSimon Glass 10806811959SSimon Glass return 0; 10906811959SSimon Glass } 11006811959SSimon Glass 111b6a49a7aSSimon Glass #ifdef CONFIG_DM_TEST 11206811959SSimon Glass static int do_dm_test(cmd_tbl_t *cmdtp, int flag, int argc, 11306811959SSimon Glass char * const argv[]) 11406811959SSimon Glass { 11506811959SSimon Glass return dm_test_main(); 11606811959SSimon Glass } 117b6a49a7aSSimon Glass #define TEST_HELP "\ndm test Run tests" 118b6a49a7aSSimon Glass #else 119b6a49a7aSSimon Glass #define TEST_HELP 120b6a49a7aSSimon Glass #endif 12106811959SSimon Glass 12206811959SSimon Glass static cmd_tbl_t test_commands[] = { 12306811959SSimon Glass U_BOOT_CMD_MKENT(tree, 0, 1, do_dm_dump_all, "", ""), 12406811959SSimon Glass U_BOOT_CMD_MKENT(uclass, 1, 1, do_dm_dump_uclass, "", ""), 125b6a49a7aSSimon Glass #ifdef CONFIG_DM_TEST 12606811959SSimon Glass U_BOOT_CMD_MKENT(test, 1, 1, do_dm_test, "", ""), 127b6a49a7aSSimon Glass #endif 12806811959SSimon Glass }; 12906811959SSimon Glass 13006811959SSimon Glass static int do_dm(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[]) 13106811959SSimon Glass { 13206811959SSimon Glass cmd_tbl_t *test_cmd; 13306811959SSimon Glass int ret; 13406811959SSimon Glass 13506811959SSimon Glass if (argc != 2) 13606811959SSimon Glass return CMD_RET_USAGE; 13706811959SSimon Glass test_cmd = find_cmd_tbl(argv[1], test_commands, 13806811959SSimon Glass ARRAY_SIZE(test_commands)); 13906811959SSimon Glass argc -= 2; 14006811959SSimon Glass argv += 2; 14106811959SSimon Glass if (!test_cmd || argc > test_cmd->maxargs) 14206811959SSimon Glass return CMD_RET_USAGE; 14306811959SSimon Glass 14406811959SSimon Glass ret = test_cmd->cmd(test_cmd, flag, argc, argv); 14506811959SSimon Glass 14606811959SSimon Glass return cmd_process_error(test_cmd, ret); 14706811959SSimon Glass } 14806811959SSimon Glass 14906811959SSimon Glass U_BOOT_CMD( 15006811959SSimon Glass dm, 2, 1, do_dm, 15106811959SSimon Glass "Driver model low level access", 152fffa24d7SSimon Glass "tree Dump driver model tree ('*' = activated)\n" 153b6a49a7aSSimon Glass "dm uclass Dump list of instances for each uclass" 154b6a49a7aSSimon Glass TEST_HELP 15506811959SSimon Glass ); 156