Skip to content

Commit 6f59646

Browse files
maciej-w-rozyckigregkh
authored andcommitted
serial: dz: Convert to use a platform device
commit 5d7a49d60b8fda66da60e240fd7315232fa1754f upstream. Prevent a crash from happening as the first serial port is initialised: Console: switching to colour frame buffer device 160x64 tgafb: SFB+ detected, rev=0x02 fb0: Digital ZLX-E1 frame buffer device at 0x1e000000 DECstation DZ serial driver version 1.04 CPU 0 Unable to handle kernel paging request at virtual address 000000bc, epc == 8048b3a4, ra == 80470a78 Oops[starfive-tech#1]: CPU: 0 UID: 0 PID: 1 Comm: swapper/0 Not tainted 6.19.0-dirty starfive-tech#35 NONE $ 0 : 00000000 1000ac00 00000004 804707ac $ 4 : 00000000 80e20850 80e20858 81000030 $ 8 : 00000000 8072c81c 00000008 fefefeff $12 : 6c616972 00000006 80c5917f 69726420 $16 : 80e20800 00000000 808f8968 80e20800 $20 : 00000000 807f5a9 808b0094 808d3bc8 $24 : 00000018 80479030 $28 : 80c2e000 80c2fd70 00000069 80470a78 Hi : 00000004 Lo : 00000000 epc : 8048b3a4 __dev_fwnode+0x0/0xc ra : 80470a78 serial_base_ctrl_add+0xa0/0x168 Status: 1000ac04 IEp Cause : 30000008 (ExcCode 02) BadVA : 000000bc PrId : 00000220 (R3000) Modules linked in: Process swapper/0 (pid: 1, threadinfo=(ptrval), task=(ptrval), tls=00000000) Stack : 00400044 00400040 8046f4cc 00000000 808a6148 808a0000 808f8968 8086983c 808e0000 8046fc84 1000ac01 00000028 80e20700 802ba3f8 80e20700 80d34a94 80c1b900 80e20700 80e20700 80e20700 80e20700 80444650 00000000 00000000 00000000 807f5a9 808b0094 80447080 00400040 808e0000 80d34a94 808a6148 80d34a94 00000004 80e20700 00000000 8076974c 80469810 80c2fe3c 1000ac01 ... Call Trace: [<8048b3a4>] __dev_fwnode+0x0/0xc [<80470a78>] serial_base_ctrl_add+0xa0/0x168 [<8046fc84>] serial_core_register_port+0x1c8/0x974 [<808c6af0>] dz_init+0x74/0xc8 [<800470e0>] do_one_initcall+0x44/0x2d4 [<808b111c>] kernel_init_freeable+0x258/0x308 [<8072e434>] kernel_init+0x20/0x114 [<80049cd0>] ret_from_kernel_thread+0x14/0x1c Code: 27bd0018 03e00008 2402ffea <8c8200bc> 03e00008 00000000 27bdffc0 afbe0038 afb30024 ---[ end trace 0000000000000000 ]--- -- where a pointer is dereferenced that has been derived from a null pointer to the port's parent device. Since no device is available with legacy probing and it's not anymore a preferable way to discover devices anyway, switch the driver to using a platform device and use it as the port's parent device. Update resource handling accordingly and only request the actual span of addresses used within the slot, which will have had its resource already requested by generic platform device code. Use platform_driver_probe() not just because the DZ device is fixed with solder on board and not straightforward to remove, but foremost because the associated TTY's major device number is the same as used by the zs driver and the first driver to claim it will prevent the other one from using it. Either one DZ device or some SCC devices will be present in a given system but never both at a time, and therefore we want the major device number to be claimed by the first driver to actually successfully bind to its device and platform_driver_probe() is a way to fulfil that. An unfortunate consequence of the switch to a platform device is we now hand the console over from the bootconsole much later in the bootstrap. The firmware console handler appears good enough though to work so late and in particular with interrupts enabled. Conversely only starting the console port so late lets the reset code fully utilise our delay handlers, so switch from udelay() to fsleep() for transmitter draining so as to avoid busy-waiting for an excessive amount of time. Fixes: 84a9582 ("serial: core: Start managing serial controllers to enable runtime PM") Signed-off-by: Maciej W. Rozycki <macro@orcam.me.uk> Cc: stable@vger.kernel.org # needs to use .remove_new for <= 6.10 Link: https://patch.msgid.link/alpine.DEB.2.21.2605062326540.46195@angie.orcam.me.uk Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
1 parent a3515bb commit 6f59646

2 files changed

Lines changed: 110 additions & 61 deletions

File tree

arch/mips/dec/platform.c

Lines changed: 54 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,13 @@
1010
#include <linux/mc146818rtc.h>
1111
#include <linux/platform_device.h>
1212

13+
#include <asm/bootinfo.h>
14+
15+
#include <asm/dec/interrupts.h>
16+
#include <asm/dec/kn01.h>
17+
#include <asm/dec/kn02.h>
18+
#include <asm/dec/system.h>
19+
1320
static struct resource dec_rtc_resources[] = {
1421
{
1522
.name = "rtc",
@@ -30,11 +37,57 @@ static struct platform_device dec_rtc_device = {
3037
.num_resources = ARRAY_SIZE(dec_rtc_resources),
3138
};
3239

40+
static struct resource dec_dz_resources[] = {
41+
{ .name = "dz", .flags = IORESOURCE_MEM, },
42+
{ .name = "dz", .flags = IORESOURCE_IRQ, },
43+
};
44+
45+
static struct platform_device dec_dz_device = {
46+
.name = "dz",
47+
.id = PLATFORM_DEVID_NONE,
48+
.resource = dec_dz_resources,
49+
.num_resources = ARRAY_SIZE(dec_dz_resources),
50+
};
51+
52+
static struct platform_device *dec_dz_devices[] __initdata = {
53+
&dec_dz_device,
54+
};
55+
3356
static int __init dec_add_devices(void)
3457
{
58+
int ret1, ret2;
59+
int num_dz;
60+
int irq, i;
61+
3562
dec_rtc_resources[0].start = RTC_PORT(0);
3663
dec_rtc_resources[0].end = RTC_PORT(0) + dec_kn_slot_size - 1;
37-
return platform_device_register(&dec_rtc_device);
64+
65+
i = 0;
66+
irq = dec_interrupt[DEC_IRQ_DZ11];
67+
if (IS_ENABLED(CONFIG_32BIT) && irq >= 0) {
68+
resource_size_t base;
69+
70+
switch (mips_machtype) {
71+
case MACH_DS23100:
72+
case MACH_DS5100:
73+
base = dec_kn_slot_base + KN01_DZ11;
74+
break;
75+
default:
76+
base = dec_kn_slot_base + KN02_DZ11;
77+
break;
78+
}
79+
dec_dz_device.resource[0].start = base;
80+
dec_dz_device.resource[0].end = base + dec_kn_slot_size - 1;
81+
dec_dz_device.resource[1].start = irq;
82+
dec_dz_device.resource[1].end = irq;
83+
i++;
84+
}
85+
num_dz = i;
86+
87+
ret1 = platform_device_register(&dec_rtc_device);
88+
ret2 = IS_ENABLED(CONFIG_32BIT) ?
89+
platform_add_devices(dec_dz_devices, num_dz) : 0;
90+
return ret1 ? ret1 : ret2;
3891
}
3992

4093
device_initcall(dec_add_devices);

drivers/tty/serial/dz.c

Lines changed: 56 additions & 60 deletions
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@
4040
#include <linux/kernel.h>
4141
#include <linux/major.h>
4242
#include <linux/module.h>
43+
#include <linux/platform_device.h>
4344
#include <linux/serial.h>
4445
#include <linux/serial_core.h>
4546
#include <linux/sysrq.h>
@@ -48,14 +49,6 @@
4849

4950
#include <linux/atomic.h>
5051
#include <linux/io.h>
51-
#include <asm/bootinfo.h>
52-
53-
#include <asm/dec/interrupts.h>
54-
#include <asm/dec/kn01.h>
55-
#include <asm/dec/kn02.h>
56-
#include <asm/dec/machtype.h>
57-
#include <asm/dec/prom.h>
58-
#include <asm/dec/system.h>
5952

6053
#include "dz.h"
6154

@@ -65,7 +58,9 @@ MODULE_LICENSE("GPL");
6558

6659

6760
static char dz_name[] __initdata = "DECstation DZ serial driver version ";
68-
static char dz_version[] __initdata = "1.04";
61+
static char dz_version[] __initdata = "1.05";
62+
63+
#define DZ_IO_SIZE 0x20 /* IOMEM space size. */
6964

7065
struct dz_port {
7166
struct dz_mux *mux;
@@ -81,6 +76,7 @@ struct dz_mux {
8176
};
8277

8378
static struct dz_mux dz_mux;
79+
static struct uart_driver dz_reg;
8480

8581
static inline struct dz_port *to_dport(struct uart_port *uport)
8682
{
@@ -564,7 +560,7 @@ static void dz_reset(struct dz_port *dport)
564560
iob();
565561
udelay(2); /* 1.4us TRDY recovery. */
566562
}
567-
udelay(1200); /* Transmitter drain. */
563+
fsleep(1200); /* Transmitter drain. */
568564
}
569565

570566
dz_out(dport, DZ_CSR, DZ_CLR);
@@ -681,14 +677,13 @@ static void dz_release_port(struct uart_port *uport)
681677

682678
map_guard = atomic_add_return(-1, &mux->map_guard);
683679
if (!map_guard)
684-
release_mem_region(uport->mapbase, dec_kn_slot_size);
680+
release_mem_region(uport->mapbase, DZ_IO_SIZE);
685681
}
686682

687683
static int dz_map_port(struct uart_port *uport)
688684
{
689685
if (!uport->membase)
690-
uport->membase = ioremap(uport->mapbase,
691-
dec_kn_slot_size);
686+
uport->membase = ioremap(uport->mapbase, DZ_IO_SIZE);
692687
if (!uport->membase) {
693688
printk(KERN_ERR "dz: Cannot map MMIO\n");
694689
return -ENOMEM;
@@ -704,8 +699,7 @@ static int dz_request_port(struct uart_port *uport)
704699

705700
map_guard = atomic_add_return(1, &mux->map_guard);
706701
if (map_guard == 1) {
707-
if (!request_mem_region(uport->mapbase, dec_kn_slot_size,
708-
"dz")) {
702+
if (!request_mem_region(uport->mapbase, DZ_IO_SIZE, "dz")) {
709703
atomic_add(-1, &mux->map_guard);
710704
printk(KERN_ERR
711705
"dz: Unable to reserve MMIO resource\n");
@@ -716,7 +710,7 @@ static int dz_request_port(struct uart_port *uport)
716710
if (ret) {
717711
map_guard = atomic_add_return(-1, &mux->map_guard);
718712
if (!map_guard)
719-
release_mem_region(uport->mapbase, dec_kn_slot_size);
713+
release_mem_region(uport->mapbase, DZ_IO_SIZE);
720714
return ret;
721715
}
722716
return 0;
@@ -768,35 +762,49 @@ static const struct uart_ops dz_ops = {
768762
.verify_port = dz_verify_port,
769763
};
770764

771-
static void __init dz_init_ports(void)
765+
static int __init dz_probe(struct platform_device *pdev)
772766
{
773-
static int first = 1;
774-
unsigned long base;
767+
struct resource *mem_resource, *irq_resource;
775768
int line;
776769

777-
if (!first)
778-
return;
779-
first = 0;
780-
781-
if (mips_machtype == MACH_DS23100 || mips_machtype == MACH_DS5100)
782-
base = dec_kn_slot_base + KN01_DZ11;
783-
else
784-
base = dec_kn_slot_base + KN02_DZ11;
770+
mem_resource = platform_get_resource(pdev, IORESOURCE_MEM, 0);
771+
irq_resource = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
772+
if (!mem_resource || !irq_resource)
773+
return -ENODEV;
785774

786775
for (line = 0; line < DZ_NB_PORT; line++) {
787776
struct dz_port *dport = &dz_mux.dport[line];
788777
struct uart_port *uport = &dport->port;
789778

790779
dport->mux = &dz_mux;
791780

792-
uport->irq = dec_interrupt[DEC_IRQ_DZ11];
781+
uport->dev = &pdev->dev;
782+
uport->irq = irq_resource->start;
793783
uport->fifosize = 1;
794784
uport->iotype = UPIO_MEM;
795785
uport->flags = UPF_BOOT_AUTOCONF;
796786
uport->ops = &dz_ops;
797787
uport->line = line;
798-
uport->mapbase = base;
788+
uport->mapbase = mem_resource->start;
799789
uport->has_sysrq = IS_ENABLED(CONFIG_SERIAL_DZ_CONSOLE);
790+
791+
if (uart_add_one_port(&dz_reg, uport))
792+
uport->dev = NULL;
793+
}
794+
795+
return 0;
796+
}
797+
798+
static void __exit dz_remove(struct platform_device *pdev)
799+
{
800+
int line;
801+
802+
for (line = DZ_NB_PORT - 1; line >= 0; line--) {
803+
struct dz_port *dport = &dz_mux.dport[line];
804+
struct uart_port *uport = &dport->port;
805+
806+
if (uport->dev)
807+
uart_remove_one_port(&dz_reg, uport);
800808
}
801809
}
802810

@@ -879,21 +887,14 @@ static int __init dz_console_setup(struct console *co, char *options)
879887
int bits = 8;
880888
int parity = 'n';
881889
int flow = 'n';
882-
int ret;
883-
884-
ret = dz_map_port(uport);
885-
if (ret)
886-
return ret;
887-
888-
dz_reset(dport);
889890

891+
if (!dport->mux)
892+
return -ENODEV;
890893
if (options)
891894
uart_parse_options(options, &baud, &parity, &bits, &flow);
892-
893-
return uart_set_options(&dport->port, co, baud, parity, bits, flow);
895+
return uart_set_options(uport, co, baud, parity, bits, flow);
894896
}
895897

896-
static struct uart_driver dz_reg;
897898
static struct console dz_console = {
898899
.name = "ttyS",
899900
.write = dz_console_print,
@@ -904,18 +905,6 @@ static struct console dz_console = {
904905
.data = &dz_reg,
905906
};
906907

907-
static int __init dz_serial_console_init(void)
908-
{
909-
if (!IOASIC) {
910-
dz_init_ports();
911-
register_console(&dz_console);
912-
return 0;
913-
} else
914-
return -ENXIO;
915-
}
916-
917-
console_initcall(dz_serial_console_init);
918-
919908
#define SERIAL_DZ_CONSOLE &dz_console
920909
#else
921910
#define SERIAL_DZ_CONSOLE NULL
@@ -931,25 +920,32 @@ static struct uart_driver dz_reg = {
931920
.cons = SERIAL_DZ_CONSOLE,
932921
};
933922

923+
static struct platform_driver dz_driver = {
924+
.remove = __exit_p(dz_remove),
925+
.driver = { .name = "dz" },
926+
};
927+
934928
static int __init dz_init(void)
935929
{
936-
int ret, i;
937-
938-
if (IOASIC)
939-
return -ENXIO;
930+
int ret;
940931

941932
printk("%s%s\n", dz_name, dz_version);
942933

943-
dz_init_ports();
944-
945934
ret = uart_register_driver(&dz_reg);
946935
if (ret)
947936
return ret;
937+
ret = platform_driver_probe(&dz_driver, dz_probe);
938+
if (ret)
939+
uart_unregister_driver(&dz_reg);
948940

949-
for (i = 0; i < DZ_NB_PORT; i++)
950-
uart_add_one_port(&dz_reg, &dz_mux.dport[i].port);
941+
return ret;
942+
}
951943

952-
return 0;
944+
static void __exit dz_exit(void)
945+
{
946+
platform_driver_unregister(&dz_driver);
947+
uart_unregister_driver(&dz_reg);
953948
}
954949

955950
module_init(dz_init);
951+
module_exit(dz_exit);

0 commit comments

Comments
 (0)