blob: 981e6f0b60f9e747e91c2fcc1628dc0358df2810 [file] [log] [blame]
Felix Held3c44c622022-01-10 20:57:29 +01001/* SPDX-License-Identifier: GPL-2.0-only */
2
Felix Held3c44c622022-01-10 20:57:29 +01003#include <amdblocks/aoac.h>
4#include <amdblocks/gpio.h>
5#include <amdblocks/uart.h>
6#include <commonlib/helpers.h>
7#include <console/console.h>
8#include <device/device.h>
9#include <device/mmio.h>
10#include <soc/aoac_defs.h>
11#include <soc/gpio.h>
12#include <soc/southbridge.h>
13#include <soc/uart.h>
14#include <types.h>
15
16static const struct {
17 uintptr_t base;
18 struct soc_amd_gpio mux[2];
19} uart_info[] = {
20 [0] = { APU_UART0_BASE, {
21 PAD_NF(GPIO_143, UART0_TXD, PULL_NONE),
22 PAD_NF(GPIO_141, UART0_RXD, PULL_NONE),
23 } },
24 [1] = { APU_UART1_BASE, {
25 PAD_NF(GPIO_140, UART1_TXD, PULL_NONE),
26 PAD_NF(GPIO_142, UART1_RXD, PULL_NONE),
27 } },
Felix Heldb1fe9de2022-01-12 23:18:54 +010028 [2] = { APU_UART2_BASE, {
29 PAD_NF(GPIO_138, UART2_TXD, PULL_NONE),
30 PAD_NF(GPIO_136, UART2_RXD, PULL_NONE),
31 } },
32 [3] = { APU_UART3_BASE, {
33 PAD_NF(GPIO_135, UART3_TXD, PULL_NONE),
34 PAD_NF(GPIO_137, UART3_RXD, PULL_NONE),
35 } },
36 [4] = { APU_UART4_BASE, {
37 PAD_NF(GPIO_156, UART4_TXD, PULL_NONE),
38 PAD_NF(GPIO_155, UART4_RXD, PULL_NONE),
39 } },
Felix Held3c44c622022-01-10 20:57:29 +010040};
41
42uintptr_t get_uart_base(unsigned int idx)
43{
44 if (idx >= ARRAY_SIZE(uart_info))
45 return 0;
46
47 return uart_info[idx].base;
48}
49
50void clear_uart_legacy_config(void)
51{
52 write16((void *)FCH_LEGACY_UART_DECODE, 0);
53}
54
55void set_uart_config(unsigned int idx)
56{
57 if (idx >= ARRAY_SIZE(uart_info))
58 return;
59
60 gpio_configure_pads(uart_info[idx].mux, 2);
61}
62
63static const char *uart_acpi_name(const struct device *dev)
64{
65 switch (dev->path.mmio.addr) {
66 case APU_UART0_BASE:
67 return "FUR0";
68 case APU_UART1_BASE:
69 return "FUR1";
Felix Heldb1fe9de2022-01-12 23:18:54 +010070 case APU_UART2_BASE:
71 return "FUR2";
72 case APU_UART3_BASE:
73 return "FUR3";
74 case APU_UART4_BASE:
75 return "FUR4";
Felix Held3c44c622022-01-10 20:57:29 +010076 default:
77 return NULL;
78 }
79}
80
81/* Even though this is called enable, it gets called for both enabled and disabled devices. */
82static void uart_enable(struct device *dev)
83{
84 unsigned int dev_id;
85
86 switch (dev->path.mmio.addr) {
87 case APU_UART0_BASE:
88 dev_id = FCH_AOAC_DEV_UART0;
89 break;
90 case APU_UART1_BASE:
91 dev_id = FCH_AOAC_DEV_UART1;
92 break;
Felix Heldb1fe9de2022-01-12 23:18:54 +010093 case APU_UART2_BASE:
94 dev_id = FCH_AOAC_DEV_UART2;
95 break;
96 case APU_UART3_BASE:
97 dev_id = FCH_AOAC_DEV_UART3;
98 break;
99 case APU_UART4_BASE:
100 dev_id = FCH_AOAC_DEV_UART4;
101 break;
Felix Held3c44c622022-01-10 20:57:29 +0100102 default:
103 printk(BIOS_ERR, "%s: Unknown device: %s\n", __func__, dev_path(dev));
104 return;
105 }
106
107 if (dev->enabled) {
108 power_on_aoac_device(dev_id);
109 wait_for_aoac_enabled(dev_id);
110 } else {
111 power_off_aoac_device(dev_id);
112 }
113}
114
115static void uart_read_resources(struct device *dev)
116{
Kyösti Mälkki27d62992022-05-24 20:25:58 +0300117 mmio_resource_kb(dev, 0, dev->path.mmio.addr / KiB, 4);
Felix Held3c44c622022-01-10 20:57:29 +0100118}
119
120struct device_operations sabrina_uart_mmio_ops = {
121 .read_resources = uart_read_resources,
122 .set_resources = noop_set_resources,
123 .scan_bus = scan_static_bus,
124 .enable = uart_enable,
125 .acpi_name = uart_acpi_name,
126 .acpi_fill_ssdt = uart_inject_ssdt,
127};