summaryrefslogtreecommitdiffstats
path: root/src/core/fdtcon.c
blob: 3dc71af6d099c0a72896d57ccb5837fe90b7cd44 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
/*
 * Copyright (C) 2025 Michael Brown <mbrown@fensystems.co.uk>.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
 * published by the Free Software Foundation; either version 2 of the
 * License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
 * 02110-1301, USA.
 *
 * You can also choose to distribute this program under the terms of
 * the Unmodified Binary Distribution Licence (as given in the file
 * COPYING.UBDL), provided that you have satisfied its requirements.
 */

FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );

#include <string.h>
#include <ipxe/serial.h>
#include <ipxe/devtree.h>
#include <ipxe/fdt.h>

/** @file
 *
 * Flattened Device Tree serial console
 *
 */

#ifdef SERIAL_FDT
#define SERIAL_PREFIX_fdt
#else
#define SERIAL_PREFIX_fdt __fdt_
#endif

/** FDT console parent device */
static struct device fdtcon_parent = {
	.name = "fdtcon",
	.siblings = LIST_HEAD_INIT ( fdtcon_parent.siblings ),
	.children = LIST_HEAD_INIT ( fdtcon_parent.children ),
};

/** Colour for debug messages */
#define colour &fdtcon_parent

/**
 * Identify default serial console
 *
 * @ret uart		Default serial console UART, or NULL
 */
static struct uart * fdtcon_default ( void ) {
	unsigned int chosen;
	unsigned int stdout;
	const char *path;
	struct uart *prev;
	struct uart *uart;
	int rc;

	/* Record existing UART, if any */
	prev = list_last_entry ( &uarts, struct uart, list );

	/* Locate "/chosen" node */
	if ( ( rc = fdt_path ( &sysfdt, "/chosen", &chosen ) ) != 0 ) {
		DBGC ( colour, "FDTCON could not locate \"/chosen\": %s\n",
		       strerror ( rc ) );
		return NULL;
	}

	/* Get console device path (or alias) */
	path = fdt_string ( &sysfdt, chosen, "stdout-path" );
	if ( ! path ) {
		DBGC ( colour, "FDTCON has no console device\n" );
		return NULL;
	}
	DBGC ( colour, "FDTCON console device is \"%s\"\n", path );

	/* Locate device */
	if ( ( ( rc = fdt_path ( &sysfdt, path, &stdout ) ) != 0 ) &&
	     ( ( rc = fdt_alias ( &sysfdt, path, &stdout ) ) != 0 ) ) {
		DBGC ( colour, "FDTCON could not locate \"/%s\": %s\n",
		       path, strerror ( rc ) );
		return NULL;
	}

	/* Probe device */
	if ( ( rc = dt_probe_node ( &fdtcon_parent, stdout ) ) != 0 ) {
		DBGC ( colour, "FDTCON could not probe \"%s\": %s\n",
		       path, strerror ( rc ) );
		return NULL;
	}

	/* Use newly added UART, if any */
	uart = list_last_entry ( &uarts, struct uart, list );
	if ( uart == prev )
		return NULL;

	DBGC ( colour, "FDTCON using UART %s\n", uart->name );
	return uart;
}

PROVIDE_SERIAL ( fdt, default_serial_console, fdtcon_default );