[PATCH] serial: 8250_bcm2835aux: Add ACPI support
Jeremy Linton
jeremy.linton at arm.com
Tue Feb 1 12:42:08 PST 2022
Hi,
On 2/1/22 13:24, Florian Fainelli wrote:
>
>
> On 2/1/2022 10:50 AM, Adrien Thierry wrote:
>> Add ACPI support to 8250_bcm2835aux driver. This makes it possible to
>> use the miniuart on the Raspberry Pi with the tianocore/edk2 UEFI
>> firmware.
>>
>> Signed-off-by: Adrien Thierry <athierry at redhat.com>
>> ---
>> drivers/tty/serial/8250/8250_bcm2835aux.c | 103 +++++++++++++++++-----
>> 1 file changed, 83 insertions(+), 20 deletions(-)
>>
>> diff --git a/drivers/tty/serial/8250/8250_bcm2835aux.c
>> b/drivers/tty/serial/8250/8250_bcm2835aux.c
>> index fd95860cd..b904b321e 100644
>> --- a/drivers/tty/serial/8250/8250_bcm2835aux.c
>> +++ b/drivers/tty/serial/8250/8250_bcm2835aux.c
>> @@ -12,6 +12,7 @@
>> * simultaneously to rs485.
>> */
>> +#include <linux/acpi.h>
>> #include <linux/clk.h>
>> #include <linux/io.h>
>> #include <linux/module.h>
>> @@ -44,6 +45,10 @@ struct bcm2835aux_data {
>> u32 cntl;
>> };
>> +struct bcm2835_aux_serial_acpi_driver_data {
>> + resource_size_t offset;
>> +};
>> +
>> static void bcm2835aux_rs485_start_tx(struct uart_8250_port *up)
>> {
>> if (!(up->port.rs485.flags & SER_RS485_RX_DURING_TX)) {
>> @@ -82,8 +87,12 @@ static int bcm2835aux_serial_probe(struct
>> platform_device *pdev)
>> {
>> struct uart_8250_port up = { };
>> struct bcm2835aux_data *data;
>> + struct bcm2835_aux_serial_acpi_driver_data *acpi_data;
>> struct resource *res;
>> int ret;
>> + resource_size_t mapbase;
>> + resource_size_t mapsize;
>> + unsigned int uartclk;
>> /* allocate the custom structure */
>> data = devm_kzalloc(&pdev->dev, sizeof(*data), GFP_KERNEL);
>> @@ -108,10 +117,12 @@ static int bcm2835aux_serial_probe(struct
>> platform_device *pdev)
>> platform_set_drvdata(pdev, data);
>> - /* get the clock - this also enables the HW */
>> - data->clk = devm_clk_get(&pdev->dev, NULL);
>> - if (IS_ERR(data->clk))
>> - return dev_err_probe(&pdev->dev, PTR_ERR(data->clk), "could
>> not get clk\n");
>> + if (dev_of_node(&pdev->dev)) {
>> + /* get the clock - this also enables the HW */
>> + data->clk = devm_clk_get(&pdev->dev, NULL);
>> + if (IS_ERR(data->clk))
>> + return dev_err_probe(&pdev->dev, PTR_ERR(data->clk),
>> "could not get clk\n");
>> + }
>
> This does not seem necessary, if the clk is NULL when probed via ACPI,
> all of the clk_* APIs will deal with that gracefully. If you need not to
> treat -ENOENT as a hard error here, consider switching to
> devm_clk_get_optional(). Given that you look at the 'clock-frequency'
> property, you can still have some generic code, something like:
>
> if (IS_ERR(data->clk)) {
> ret = device_property_read_u32(&pdev->dev, "clock-frequency",
> &uartclk);
> if (ret)
> return dev_err_probe(&pdev->dev, ret, "could not get clk\n");
> }
>
>> /* get the interrupt */
>> ret = platform_get_irq(pdev, 0);
>> @@ -125,20 +136,59 @@ static int bcm2835aux_serial_probe(struct
>> platform_device *pdev)
>> dev_err(&pdev->dev, "memory resource not found");
>> return -EINVAL;
>> }
>> - up.port.mapbase = res->start;
>> - up.port.mapsize = resource_size(res);
>> -
>> - /* Check for a fixed line number */
>> - ret = of_alias_get_id(pdev->dev.of_node, "serial");
>> - if (ret >= 0)
>> - up.port.line = ret;
>> -
>> - /* enable the clock as a last step */
>> - ret = clk_prepare_enable(data->clk);
>> - if (ret) {
>> - dev_err(&pdev->dev, "unable to enable uart clock - %d\n",
>> - ret);
>> - return ret;
>
> All of that path can be common, and you can just define an offset to
> apply to the resource at the top after you fetched the memory resource.
> The offset will be non-0 for ACPI and 0 for non-ACPI. That is, no need
> for the intermediate variables and conditional paths whether this is
> ACPI apply this offset, or not.
>
>> +
>> + mapbase = res->start;
>> + mapsize = resource_size(res);
>> +
>> + if (has_acpi_companion(&pdev->dev)) {
>> + const struct acpi_device_id *match;
>> +
>> + match = acpi_match_device(pdev->dev.driver->acpi_match_table,
>> &pdev->dev);
>> + if (!match)
>> + return -ENODEV;
>> +
>> + acpi_data = (struct bcm2835_aux_serial_acpi_driver_data
>> *)match->driver_data;
>> +
>> + /* Some UEFI implementations (e.g. tianocore/edk2 for the
>> Raspberry Pi)
>> + * describe the miniuart with a base address that encompasses
>> the auxiliary
>> + * registers shared between the miniuart and spi.
>> + *
>> + * This is due to historical reasons, see discussion here :
>> + * https://edk2.groups.io/g/devel/topic/87501357#84349
>> + *
>> + * We need to add the offset between the miniuart and auxiliary
>> + * registers to get the real miniuart base address.
>
> And ACPI on the Pi4 is so widely deployed that fixing the miniuart
> resources is not an option at all? This really really continues to
> contribute to my impression that ACPI on the Pi4 is a fad more than a
> real thing, sorry.
The problem again, is that this resource is legacy and used by
windows/vmware/etc on both the rpi3 and rpi4. So, unfortunately it
cannot really be changed without breaking existing OSs.
Thanks,
More information about the linux-arm-kernel
mailing list