blob: 8b6ce11a63b5b5df293bc38b961b98e18ac99559 [file] [log] [blame]
David Wue7ae4cf2019-01-02 21:00:55 +08001// SPDX-License-Identifier: GPL-2.0+
2/*
3 * (C) Copyright 2019 Rockchip Electronics Co., Ltd
4 */
5
6#include <common.h>
7#include <dm.h>
8#include <dm/pinctrl.h>
9#include <regmap.h>
10#include <syscon.h>
11
12#include "pinctrl-rockchip.h"
13
14static struct rockchip_mux_route_data rk3288_mux_route_data[] = {
15 {
16 /* edphdmi_cecinoutt1 */
17 .bank_num = 7,
18 .pin = 16,
19 .func = 2,
20 .route_offset = 0x264,
21 .route_val = BIT(16 + 12) | BIT(12),
22 }, {
23 /* edphdmi_cecinout */
24 .bank_num = 7,
25 .pin = 23,
26 .func = 4,
27 .route_offset = 0x264,
28 .route_val = BIT(16 + 12),
29 },
30};
31
32#define RK3288_PULL_OFFSET 0x140
33#define RK3288_PULL_PMU_OFFSET 0x64
34
35static void rk3288_calc_pull_reg_and_bit(struct rockchip_pin_bank *bank,
36 int pin_num, struct regmap **regmap,
37 int *reg, u8 *bit)
38{
39 struct rockchip_pinctrl_priv *priv = bank->priv;
40
41 /* The first 24 pins of the first bank are located in PMU */
42 if (bank->bank_num == 0) {
43 *regmap = priv->regmap_pmu;
44 *reg = RK3288_PULL_PMU_OFFSET;
45
46 *reg += ((pin_num / ROCKCHIP_PULL_PINS_PER_REG) * 4);
47 *bit = pin_num % ROCKCHIP_PULL_PINS_PER_REG;
48 *bit *= ROCKCHIP_PULL_BITS_PER_PIN;
49 } else {
50 *regmap = priv->regmap_base;
51 *reg = RK3288_PULL_OFFSET;
52
53 /* correct the offset, as we're starting with the 2nd bank */
54 *reg -= 0x10;
55 *reg += bank->bank_num * ROCKCHIP_PULL_BANK_STRIDE;
56 *reg += ((pin_num / ROCKCHIP_PULL_PINS_PER_REG) * 4);
57
58 *bit = (pin_num % ROCKCHIP_PULL_PINS_PER_REG);
59 *bit *= ROCKCHIP_PULL_BITS_PER_PIN;
60 }
61}
62
63#define RK3288_DRV_PMU_OFFSET 0x70
64#define RK3288_DRV_GRF_OFFSET 0x1c0
65
66static void rk3288_calc_drv_reg_and_bit(struct rockchip_pin_bank *bank,
67 int pin_num, struct regmap **regmap,
68 int *reg, u8 *bit)
69{
70 struct rockchip_pinctrl_priv *priv = bank->priv;
71
72 /* The first 24 pins of the first bank are located in PMU */
73 if (bank->bank_num == 0) {
74 *regmap = priv->regmap_pmu;
75 *reg = RK3288_DRV_PMU_OFFSET;
76
77 *reg += ((pin_num / ROCKCHIP_DRV_PINS_PER_REG) * 4);
78 *bit = pin_num % ROCKCHIP_DRV_PINS_PER_REG;
79 *bit *= ROCKCHIP_DRV_BITS_PER_PIN;
80 } else {
81 *regmap = priv->regmap_base;
82 *reg = RK3288_DRV_GRF_OFFSET;
83
84 /* correct the offset, as we're starting with the 2nd bank */
85 *reg -= 0x10;
86 *reg += bank->bank_num * ROCKCHIP_DRV_BANK_STRIDE;
87 *reg += ((pin_num / ROCKCHIP_DRV_PINS_PER_REG) * 4);
88
89 *bit = (pin_num % ROCKCHIP_DRV_PINS_PER_REG);
90 *bit *= ROCKCHIP_DRV_BITS_PER_PIN;
91 }
92}
93
94static struct rockchip_pin_bank rk3288_pin_banks[] = {
David Wu50298092019-02-12 19:51:51 +080095 PIN_BANK_IOMUX_DRV_PULL_FLAGS(0, 24, "gpio0",
96 IOMUX_SOURCE_PMU | IOMUX_WRITABLE_32BIT,
97 IOMUX_SOURCE_PMU | IOMUX_WRITABLE_32BIT,
98 IOMUX_SOURCE_PMU | IOMUX_WRITABLE_32BIT,
99 IOMUX_UNROUTED,
100 DRV_TYPE_WRITABLE_32BIT,
101 DRV_TYPE_WRITABLE_32BIT,
102 DRV_TYPE_WRITABLE_32BIT,
103 0,
104 PULL_TYPE_WRITABLE_32BIT,
105 PULL_TYPE_WRITABLE_32BIT,
106 PULL_TYPE_WRITABLE_32BIT,
107 0
David Wue7ae4cf2019-01-02 21:00:55 +0800108 ),
109 PIN_BANK_IOMUX_FLAGS(1, 32, "gpio1", IOMUX_UNROUTED,
110 IOMUX_UNROUTED,
111 IOMUX_UNROUTED,
112 0
113 ),
114 PIN_BANK_IOMUX_FLAGS(2, 32, "gpio2", 0, 0, 0, IOMUX_UNROUTED),
115 PIN_BANK_IOMUX_FLAGS(3, 32, "gpio3", 0, 0, 0, IOMUX_WIDTH_4BIT),
116 PIN_BANK_IOMUX_FLAGS(4, 32, "gpio4", IOMUX_WIDTH_4BIT,
117 IOMUX_WIDTH_4BIT,
118 0,
119 0
120 ),
121 PIN_BANK_IOMUX_FLAGS(5, 32, "gpio5", IOMUX_UNROUTED,
122 0,
123 0,
124 IOMUX_UNROUTED
125 ),
126 PIN_BANK_IOMUX_FLAGS(6, 32, "gpio6", 0, 0, 0, IOMUX_UNROUTED),
127 PIN_BANK_IOMUX_FLAGS(7, 32, "gpio7", 0,
128 0,
129 IOMUX_WIDTH_4BIT,
130 IOMUX_UNROUTED
131 ),
132 PIN_BANK(8, 16, "gpio8"),
133};
134
135static struct rockchip_pin_ctrl rk3288_pin_ctrl = {
136 .pin_banks = rk3288_pin_banks,
137 .nr_banks = ARRAY_SIZE(rk3288_pin_banks),
138 .label = "RK3288-GPIO",
139 .type = RK3288,
140 .grf_mux_offset = 0x0,
141 .pmu_mux_offset = 0x84,
142 .iomux_routes = rk3288_mux_route_data,
143 .niomux_routes = ARRAY_SIZE(rk3288_mux_route_data),
144 .pull_calc_reg = rk3288_calc_pull_reg_and_bit,
145 .drv_calc_reg = rk3288_calc_drv_reg_and_bit,
146};
147
148static const struct udevice_id rk3288_pinctrl_ids[] = {
149 {
150 .compatible = "rockchip,rk3288-pinctrl",
151 .data = (ulong)&rk3288_pin_ctrl
152 },
153 { }
154};
155
156U_BOOT_DRIVER(pinctrl_rk3288) = {
157 .name = "rockchip_rk3288_pinctrl",
158 .id = UCLASS_PINCTRL,
159 .of_match = rk3288_pinctrl_ids,
160 .priv_auto_alloc_size = sizeof(struct rockchip_pinctrl_priv),
161 .ops = &rockchip_pinctrl_ops,
162#if !CONFIG_IS_ENABLED(OF_PLATDATA)
163 .bind = dm_scan_fdt_dev,
164#endif
165 .probe = rockchip_pinctrl_probe,
166};