|  | // SPDX-License-Identifier: GPL-2.0-only | 
|  | /* | 
|  | * Copyright (C) 2014 Free Electrons | 
|  | * | 
|  | * Author: Boris BREZILLON <boris.brezillon@free-electrons.com> | 
|  | * | 
|  | * Allwinner PRCM (Power/Reset/Clock Management) driver | 
|  | */ | 
|  |  | 
|  | #include <linux/mfd/core.h> | 
|  | #include <linux/init.h> | 
|  | #include <linux/of.h> | 
|  |  | 
|  | #define SUN8I_CODEC_ANALOG_BASE	0x1c0 | 
|  | #define SUN8I_CODEC_ANALOG_SIZE	0x4 | 
|  |  | 
|  | struct prcm_data { | 
|  | int nsubdevs; | 
|  | const struct mfd_cell *subdevs; | 
|  | }; | 
|  |  | 
|  | static const struct resource sun6i_a31_ar100_clk_res[] = { | 
|  | DEFINE_RES_MEM(0x0, 4) | 
|  | }; | 
|  |  | 
|  | static const struct resource sun6i_a31_apb0_clk_res[] = { | 
|  | DEFINE_RES_MEM(0xc, 4) | 
|  | }; | 
|  |  | 
|  | static const struct resource sun6i_a31_apb0_gates_clk_res[] = { | 
|  | DEFINE_RES_MEM(0x28, 4) | 
|  | }; | 
|  |  | 
|  | static const struct resource sun6i_a31_ir_clk_res[] = { | 
|  | DEFINE_RES_MEM(0x54, 4) | 
|  | }; | 
|  |  | 
|  | static const struct resource sun6i_a31_apb0_rstc_res[] = { | 
|  | DEFINE_RES_MEM(0xb0, 4) | 
|  | }; | 
|  |  | 
|  | static const struct resource sun8i_codec_analog_res[] = { | 
|  | DEFINE_RES_MEM(SUN8I_CODEC_ANALOG_BASE, SUN8I_CODEC_ANALOG_SIZE), | 
|  | }; | 
|  |  | 
|  | static const struct mfd_cell sun6i_a31_prcm_subdevs[] = { | 
|  | { | 
|  | .name = "sun6i-a31-ar100-clk", | 
|  | .of_compatible = "allwinner,sun6i-a31-ar100-clk", | 
|  | .num_resources = ARRAY_SIZE(sun6i_a31_ar100_clk_res), | 
|  | .resources = sun6i_a31_ar100_clk_res, | 
|  | }, | 
|  | { | 
|  | .name = "sun6i-a31-apb0-clk", | 
|  | .of_compatible = "allwinner,sun6i-a31-apb0-clk", | 
|  | .num_resources = ARRAY_SIZE(sun6i_a31_apb0_clk_res), | 
|  | .resources = sun6i_a31_apb0_clk_res, | 
|  | }, | 
|  | { | 
|  | .name = "sun6i-a31-apb0-gates-clk", | 
|  | .of_compatible = "allwinner,sun6i-a31-apb0-gates-clk", | 
|  | .num_resources = ARRAY_SIZE(sun6i_a31_apb0_gates_clk_res), | 
|  | .resources = sun6i_a31_apb0_gates_clk_res, | 
|  | }, | 
|  | { | 
|  | .name = "sun6i-a31-ir-clk", | 
|  | .of_compatible = "allwinner,sun4i-a10-mod0-clk", | 
|  | .num_resources = ARRAY_SIZE(sun6i_a31_ir_clk_res), | 
|  | .resources = sun6i_a31_ir_clk_res, | 
|  | }, | 
|  | { | 
|  | .name = "sun6i-a31-apb0-clock-reset", | 
|  | .of_compatible = "allwinner,sun6i-a31-clock-reset", | 
|  | .num_resources = ARRAY_SIZE(sun6i_a31_apb0_rstc_res), | 
|  | .resources = sun6i_a31_apb0_rstc_res, | 
|  | }, | 
|  | }; | 
|  |  | 
|  | static const struct mfd_cell sun8i_a23_prcm_subdevs[] = { | 
|  | { | 
|  | .name = "sun8i-a23-apb0-clk", | 
|  | .of_compatible = "allwinner,sun8i-a23-apb0-clk", | 
|  | .num_resources = ARRAY_SIZE(sun6i_a31_apb0_clk_res), | 
|  | .resources = sun6i_a31_apb0_clk_res, | 
|  | }, | 
|  | { | 
|  | .name = "sun6i-a31-apb0-gates-clk", | 
|  | .of_compatible = "allwinner,sun8i-a23-apb0-gates-clk", | 
|  | .num_resources = ARRAY_SIZE(sun6i_a31_apb0_gates_clk_res), | 
|  | .resources = sun6i_a31_apb0_gates_clk_res, | 
|  | }, | 
|  | { | 
|  | .name = "sun6i-a31-apb0-clock-reset", | 
|  | .of_compatible = "allwinner,sun6i-a31-clock-reset", | 
|  | .num_resources = ARRAY_SIZE(sun6i_a31_apb0_rstc_res), | 
|  | .resources = sun6i_a31_apb0_rstc_res, | 
|  | }, | 
|  | { | 
|  | .name		= "sun8i-codec-analog", | 
|  | .of_compatible	= "allwinner,sun8i-a23-codec-analog", | 
|  | .num_resources	= ARRAY_SIZE(sun8i_codec_analog_res), | 
|  | .resources	= sun8i_codec_analog_res, | 
|  | }, | 
|  | }; | 
|  |  | 
|  | static const struct prcm_data sun6i_a31_prcm_data = { | 
|  | .nsubdevs = ARRAY_SIZE(sun6i_a31_prcm_subdevs), | 
|  | .subdevs = sun6i_a31_prcm_subdevs, | 
|  | }; | 
|  |  | 
|  | static const struct prcm_data sun8i_a23_prcm_data = { | 
|  | .nsubdevs = ARRAY_SIZE(sun8i_a23_prcm_subdevs), | 
|  | .subdevs = sun8i_a23_prcm_subdevs, | 
|  | }; | 
|  |  | 
|  | static const struct of_device_id sun6i_prcm_dt_ids[] = { | 
|  | { | 
|  | .compatible = "allwinner,sun6i-a31-prcm", | 
|  | .data = &sun6i_a31_prcm_data, | 
|  | }, | 
|  | { | 
|  | .compatible = "allwinner,sun8i-a23-prcm", | 
|  | .data = &sun8i_a23_prcm_data, | 
|  | }, | 
|  | { /* sentinel */ }, | 
|  | }; | 
|  |  | 
|  | static int sun6i_prcm_probe(struct platform_device *pdev) | 
|  | { | 
|  | const struct of_device_id *match; | 
|  | const struct prcm_data *data; | 
|  | struct resource *res; | 
|  | int ret; | 
|  |  | 
|  | match = of_match_node(sun6i_prcm_dt_ids, pdev->dev.of_node); | 
|  | if (!match) | 
|  | return -EINVAL; | 
|  |  | 
|  | data = match->data; | 
|  |  | 
|  | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 
|  | if (!res) { | 
|  | dev_err(&pdev->dev, "no prcm memory region provided\n"); | 
|  | return -ENOENT; | 
|  | } | 
|  |  | 
|  | ret = mfd_add_devices(&pdev->dev, 0, data->subdevs, data->nsubdevs, | 
|  | res, -1, NULL); | 
|  | if (ret) { | 
|  | dev_err(&pdev->dev, "failed to add subdevices\n"); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static struct platform_driver sun6i_prcm_driver = { | 
|  | .driver = { | 
|  | .name = "sun6i-prcm", | 
|  | .of_match_table = sun6i_prcm_dt_ids, | 
|  | }, | 
|  | .probe = sun6i_prcm_probe, | 
|  | }; | 
|  | builtin_platform_driver(sun6i_prcm_driver); |