}
 
 static const struct sci_port_params *
-sci_probe_regmap(const struct plat_sci_port *cfg)
+sci_probe_regmap(const struct plat_sci_port *cfg, struct sci_port *sci_port)
 {
        unsigned int regtype;
 
+       sci_port->ops = &sci_port_ops;
+       sci_port->port.ops = &sci_uart_ops;
+
        if (cfg->regtype != SCIx_PROBE_REGTYPE)
                return &sci_port_params[cfg->regtype];
 
        int ret;
 
        sci_port->cfg   = p;
-       sci_port->ops   = &sci_port_ops;
 
-       port->ops       = &sci_uart_ops;
        port->iotype    = UPIO_MEM;
        port->line      = index;
        port->has_sysrq = IS_ENABLED(CONFIG_SERIAL_SH_SCI_CONSOLE);
                for (i = 1; i < ARRAY_SIZE(sci_port->irqs); i++)
                        sci_port->irqs[i] = sci_port->irqs[0];
 
-       sci_port->params = sci_probe_regmap(p);
-       if (unlikely(sci_port->params == NULL))
-               return -EINVAL;
-
        switch (p->type) {
        case PORT_SCIFB:
                sci_port->rx_trigger = 48;
 static int sci_probe_earlyprintk(struct platform_device *pdev)
 {
        const struct plat_sci_port *cfg = dev_get_platdata(&pdev->dev);
+       struct sci_port *sp = &sci_ports[pdev->id];
 
        if (early_serial_console.data)
                return -EEXIST;
 
        early_serial_console.index = pdev->id;
 
-       sci_init_single(pdev, &sci_ports[pdev->id], pdev->id, cfg, true);
+       sp->params = sci_probe_regmap(cfg, sp);
+       if (!sp->params)
+               return -ENODEV;
+
+       sci_init_single(pdev, sp, pdev->id, cfg, true);
 
        if (!strstr(early_serial_buf, "keep"))
                early_serial_console.flags |= CON_BOOT;
                device_remove_file(&dev->dev, &dev_attr_rx_fifo_timeout);
 }
 
-#define SCI_OF_DATA(type, regtype)     (void *)((type) << 16 | (regtype))
-#define SCI_OF_TYPE(data)              ((unsigned long)(data) >> 16)
-#define SCI_OF_REGTYPE(data)           ((unsigned long)(data) & 0xffff)
+static const struct sci_of_data of_sci_scif_sh2 = {
+       .type = PORT_SCIF,
+       .regtype = SCIx_SH2_SCIF_FIFODATA_REGTYPE,
+       .ops = &sci_port_ops,
+       .uart_ops = &sci_uart_ops,
+       .params = &sci_port_params[SCIx_SH2_SCIF_FIFODATA_REGTYPE],
+};
+
+static const struct sci_of_data of_sci_scif_rz_scifa = {
+       .type = PORT_SCIF,
+       .regtype = SCIx_RZ_SCIFA_REGTYPE,
+       .ops = &sci_port_ops,
+       .uart_ops = &sci_uart_ops,
+       .params = &sci_port_params[SCIx_RZ_SCIFA_REGTYPE],
+};
+
+static const struct sci_of_data of_sci_scif_rzv2h = {
+       .type = PORT_SCIF,
+       .regtype = SCIx_RZV2H_SCIF_REGTYPE,
+       .ops = &sci_port_ops,
+       .uart_ops = &sci_uart_ops,
+       .params = &sci_port_params[SCIx_RZV2H_SCIF_REGTYPE],
+};
+
+static const struct sci_of_data of_sci_rcar_scif = {
+       .type = PORT_SCIF,
+       .regtype = SCIx_SH4_SCIF_BRG_REGTYPE,
+       .ops = &sci_port_ops,
+       .uart_ops = &sci_uart_ops,
+       .params = &sci_port_params[SCIx_SH4_SCIF_BRG_REGTYPE],
+};
+
+static const struct sci_of_data of_sci_scif_sh4 = {
+       .type = PORT_SCIF,
+       .regtype = SCIx_SH4_SCIF_REGTYPE,
+       .ops = &sci_port_ops,
+       .uart_ops = &sci_uart_ops,
+       .params = &sci_port_params[SCIx_SH4_SCIF_REGTYPE],
+};
+
+static const struct sci_of_data of_sci_scifa = {
+       .type = PORT_SCIFA,
+       .regtype = SCIx_SCIFA_REGTYPE,
+       .ops = &sci_port_ops,
+       .uart_ops = &sci_uart_ops,
+       .params = &sci_port_params[SCIx_SCIFA_REGTYPE],
+};
+
+static const struct sci_of_data of_sci_scifb = {
+       .type = PORT_SCIFB,
+       .regtype = SCIx_SCIFB_REGTYPE,
+       .ops = &sci_port_ops,
+       .uart_ops = &sci_uart_ops,
+       .params = &sci_port_params[SCIx_SCIFB_REGTYPE],
+};
+
+static const struct sci_of_data of_sci_hscif = {
+       .type = PORT_HSCIF,
+       .regtype = SCIx_HSCIF_REGTYPE,
+       .ops = &sci_port_ops,
+       .uart_ops = &sci_uart_ops,
+       .params = &sci_port_params[SCIx_HSCIF_REGTYPE],
+};
+
+static const struct sci_of_data of_sci_sci = {
+       .type = PORT_SCI,
+       .regtype = SCIx_SCI_REGTYPE,
+       .ops = &sci_port_ops,
+       .uart_ops = &sci_uart_ops,
+       .params = &sci_port_params[SCIx_SCI_REGTYPE],
+};
 
 static const struct of_device_id of_sci_match[] __maybe_unused = {
        /* SoC-specific types */
        {
                .compatible = "renesas,scif-r7s72100",
-               .data = SCI_OF_DATA(PORT_SCIF, SCIx_SH2_SCIF_FIFODATA_REGTYPE),
+               .data = &of_sci_scif_sh2,
        },
        {
                .compatible = "renesas,scif-r7s9210",
-               .data = SCI_OF_DATA(PORT_SCIF, SCIx_RZ_SCIFA_REGTYPE),
+               .data = &of_sci_scif_rz_scifa,
        },
        {
                .compatible = "renesas,scif-r9a07g044",
-               .data = SCI_OF_DATA(PORT_SCIF, SCIx_RZ_SCIFA_REGTYPE),
+               .data = &of_sci_scif_rz_scifa,
        },
        {
                .compatible = "renesas,scif-r9a09g057",
-               .data = SCI_OF_DATA(PORT_SCIF, SCIx_RZV2H_SCIF_REGTYPE),
+               .data = &of_sci_scif_rzv2h,
        },
        /* Family-specific types */
        {
                .compatible = "renesas,rcar-gen1-scif",
-               .data = SCI_OF_DATA(PORT_SCIF, SCIx_SH4_SCIF_BRG_REGTYPE),
+               .data = &of_sci_rcar_scif,
        }, {
                .compatible = "renesas,rcar-gen2-scif",
-               .data = SCI_OF_DATA(PORT_SCIF, SCIx_SH4_SCIF_BRG_REGTYPE),
+               .data = &of_sci_rcar_scif,
        }, {
                .compatible = "renesas,rcar-gen3-scif",
-               .data = SCI_OF_DATA(PORT_SCIF, SCIx_SH4_SCIF_BRG_REGTYPE),
+               .data = &of_sci_rcar_scif
        }, {
                .compatible = "renesas,rcar-gen4-scif",
-               .data = SCI_OF_DATA(PORT_SCIF, SCIx_SH4_SCIF_BRG_REGTYPE),
+               .data = &of_sci_rcar_scif
        },
        /* Generic types */
        {
                .compatible = "renesas,scif",
-               .data = SCI_OF_DATA(PORT_SCIF, SCIx_SH4_SCIF_REGTYPE),
+               .data = &of_sci_scif_sh4,
        }, {
                .compatible = "renesas,scifa",
-               .data = SCI_OF_DATA(PORT_SCIFA, SCIx_SCIFA_REGTYPE),
+               .data = &of_sci_scifa,
        }, {
                .compatible = "renesas,scifb",
-               .data = SCI_OF_DATA(PORT_SCIFB, SCIx_SCIFB_REGTYPE),
+               .data = &of_sci_scifb,
        }, {
                .compatible = "renesas,hscif",
-               .data = SCI_OF_DATA(PORT_HSCIF, SCIx_HSCIF_REGTYPE),
+               .data = &of_sci_hscif,
        }, {
                .compatible = "renesas,sci",
-               .data = SCI_OF_DATA(PORT_SCI, SCIx_SCI_REGTYPE),
+               .data = &of_sci_sci,
        }, {
                /* Terminator */
        },
        struct reset_control *rstc;
        struct plat_sci_port *p;
        struct sci_port *sp;
-       const void *data;
+       const struct sci_of_data *data;
        int id, ret;
 
        if (!IS_ENABLED(CONFIG_OF) || !np)
        sp->rstc = rstc;
        *dev_id = id;
 
-       p->type = SCI_OF_TYPE(data);
-       p->regtype = SCI_OF_REGTYPE(data);
+       p->type = data->type;
+       p->regtype = data->regtype;
+
+       sp->ops = data->ops;
+       sp->port.ops = data->uart_ops;
+       sp->params = data->params;
 
        sp->has_rtscts = of_property_read_bool(np, "uart-has-rtscts");
 
                p = sci_parse_dt(dev, &dev_id);
                if (IS_ERR(p))
                        return PTR_ERR(p);
+               sp = &sci_ports[dev_id];
        } else {
                p = dev->dev.platform_data;
                if (p == NULL) {
                }
 
                dev_id = dev->id;
+               sp = &sci_ports[dev_id];
+               sp->params = sci_probe_regmap(p, sp);
+               if (!sp->params)
+                       return -ENODEV;
        }
 
-       sp = &sci_ports[dev_id];
        sp->suspend_regs = devm_kzalloc(&dev->dev,
                                        sp->ops->suspend_regs_size(),
                                        GFP_KERNEL);
 }
 
 int __init scix_early_console_setup(struct earlycon_device *device,
-                                     int type)
+                                   const struct sci_of_data *data)
 {
        const struct sci_common_regs *regs;
 
        if (!device->port.membase)
                return -ENODEV;
 
-       device->port.type = type;
+       device->port.type = data->type;
        sci_ports[0].port = device->port;
-       port_cfg.type = type;
+
+       port_cfg.type = data->type;
+       port_cfg.regtype = data->regtype;
+
        sci_ports[0].cfg = &port_cfg;
-       sci_ports[0].ops = &sci_port_ops;
-       sci_ports[0].params = sci_probe_regmap(&port_cfg);
+       sci_ports[0].params = data->params;
+       sci_ports[0].ops = data->ops;
+       sci_ports[0].port.ops = data->uart_ops;
        sci_uart_earlycon = true;
        regs = sci_ports[0].params->common_regs;
 
 static int __init sci_early_console_setup(struct earlycon_device *device,
                                          const char *opt)
 {
-       return scix_early_console_setup(device, PORT_SCI);
+       return scix_early_console_setup(device, &of_sci_sci);
 }
 static int __init scif_early_console_setup(struct earlycon_device *device,
                                          const char *opt)
 {
-       return scix_early_console_setup(device, PORT_SCIF);
+       return scix_early_console_setup(device, &of_sci_scif_sh4);
 }
 static int __init rzscifa_early_console_setup(struct earlycon_device *device,
                                          const char *opt)
 {
-       port_cfg.regtype = SCIx_RZ_SCIFA_REGTYPE;
-       return scix_early_console_setup(device, PORT_SCIF);
+       return scix_early_console_setup(device, &of_sci_scif_rz_scifa);
 }
 
 static int __init rzv2hscif_early_console_setup(struct earlycon_device *device,
                                                const char *opt)
 {
-       port_cfg.regtype = SCIx_RZV2H_SCIF_REGTYPE;
-       return scix_early_console_setup(device, PORT_SCIF);
+       return scix_early_console_setup(device, &of_sci_scif_rzv2h);
 }
 
 static int __init scifa_early_console_setup(struct earlycon_device *device,
                                          const char *opt)
 {
-       return scix_early_console_setup(device, PORT_SCIFA);
+       return scix_early_console_setup(device, &of_sci_scifa);
 }
 static int __init scifb_early_console_setup(struct earlycon_device *device,
                                          const char *opt)
 {
-       return scix_early_console_setup(device, PORT_SCIFB);
+       return scix_early_console_setup(device, &of_sci_scifb);
 }
 static int __init hscif_early_console_setup(struct earlycon_device *device,
                                          const char *opt)
 {
-       return scix_early_console_setup(device, PORT_HSCIF);
+       return scix_early_console_setup(device, &of_sci_hscif);
 }
 
 OF_EARLYCON_DECLARE(sci, "renesas,sci", sci_early_console_setup);