1 // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
2 /*
3 * Mellanox platform driver
4 *
5 * Copyright (C) 2016-2018 Mellanox Technologies
6 * Copyright (C) 2016-2018 Vadim Pasternak <vadimp@mellanox.com>
7 */
8
9 #include <linux/device.h>
10 #include <linux/dmi.h>
11 #include <linux/i2c.h>
12 #include <linux/i2c-mux.h>
13 #include <linux/io.h>
14 #include <linux/module.h>
15 #include <linux/platform_device.h>
16 #include <linux/platform_data/i2c-mux-reg.h>
17 #include <linux/platform_data/mlxreg.h>
18 #include <linux/regmap.h>
19
20 #define MLX_PLAT_DEVICE_NAME "mlxplat"
21
22 /* LPC bus IO offsets */
23 #define MLXPLAT_CPLD_LPC_I2C_BASE_ADRR 0x2000
24 #define MLXPLAT_CPLD_LPC_REG_BASE_ADRR 0x2500
25 #define MLXPLAT_CPLD_LPC_REG_CPLD1_VER_OFFSET 0x00
26 #define MLXPLAT_CPLD_LPC_REG_CPLD2_VER_OFFSET 0x01
27 #define MLXPLAT_CPLD_LPC_REG_CPLD3_VER_OFFSET 0x02
28 #define MLXPLAT_CPLD_LPC_REG_CPLD4_VER_OFFSET 0x03
29 #define MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET 0x1d
30 #define MLXPLAT_CPLD_LPC_REG_RST_CAUSE1_OFFSET 0x1e
31 #define MLXPLAT_CPLD_LPC_REG_RST_CAUSE2_OFFSET 0x1f
32 #define MLXPLAT_CPLD_LPC_REG_LED1_OFFSET 0x20
33 #define MLXPLAT_CPLD_LPC_REG_LED2_OFFSET 0x21
34 #define MLXPLAT_CPLD_LPC_REG_LED3_OFFSET 0x22
35 #define MLXPLAT_CPLD_LPC_REG_LED4_OFFSET 0x23
36 #define MLXPLAT_CPLD_LPC_REG_LED5_OFFSET 0x24
37 #define MLXPLAT_CPLD_LPC_REG_FAN_DIRECTION 0x2a
38 #define MLXPLAT_CPLD_LPC_REG_GP1_OFFSET 0x30
39 #define MLXPLAT_CPLD_LPC_REG_WP1_OFFSET 0x31
40 #define MLXPLAT_CPLD_LPC_REG_GP2_OFFSET 0x32
41 #define MLXPLAT_CPLD_LPC_REG_WP2_OFFSET 0x33
42 #define MLXPLAT_CPLD_LPC_REG_PWM_CONTROL_OFFSET 0x37
43 #define MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET 0x3a
44 #define MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET 0x3b
45 #define MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET 0x40
46 #define MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET 0x41
47 #define MLXPLAT_CPLD_LPC_REG_AGGRCO_OFFSET 0x42
48 #define MLXPLAT_CPLD_LPC_REG_AGGRCO_MASK_OFFSET 0x43
49 #define MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET 0x50
50 #define MLXPLAT_CPLD_LPC_REG_ASIC_EVENT_OFFSET 0x51
51 #define MLXPLAT_CPLD_LPC_REG_ASIC_MASK_OFFSET 0x52
52 #define MLXPLAT_CPLD_LPC_REG_PSU_OFFSET 0x58
53 #define MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET 0x59
54 #define MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET 0x5a
55 #define MLXPLAT_CPLD_LPC_REG_PWR_OFFSET 0x64
56 #define MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET 0x65
57 #define MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET 0x66
58 #define MLXPLAT_CPLD_LPC_REG_FAN_OFFSET 0x88
59 #define MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET 0x89
60 #define MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET 0x8a
61 #define MLXPLAT_CPLD_LPC_REG_WD_CLEAR_OFFSET 0xc7
62 #define MLXPLAT_CPLD_LPC_REG_WD_CLEAR_WP_OFFSET 0xc8
63 #define MLXPLAT_CPLD_LPC_REG_WD1_TMR_OFFSET 0xc9
64 #define MLXPLAT_CPLD_LPC_REG_WD1_ACT_OFFSET 0xcb
65 #define MLXPLAT_CPLD_LPC_REG_WD2_TMR_OFFSET 0xcd
66 #define MLXPLAT_CPLD_LPC_REG_WD2_TLEFT_OFFSET 0xce
67 #define MLXPLAT_CPLD_LPC_REG_WD2_ACT_OFFSET 0xcf
68 #define MLXPLAT_CPLD_LPC_REG_WD3_TMR_OFFSET 0xd1
69 #define MLXPLAT_CPLD_LPC_REG_WD3_TLEFT_OFFSET 0xd2
70 #define MLXPLAT_CPLD_LPC_REG_WD3_ACT_OFFSET 0xd3
71 #define MLXPLAT_CPLD_LPC_REG_PWM1_OFFSET 0xe3
72 #define MLXPLAT_CPLD_LPC_REG_TACHO1_OFFSET 0xe4
73 #define MLXPLAT_CPLD_LPC_REG_TACHO2_OFFSET 0xe5
74 #define MLXPLAT_CPLD_LPC_REG_TACHO3_OFFSET 0xe6
75 #define MLXPLAT_CPLD_LPC_REG_TACHO4_OFFSET 0xe7
76 #define MLXPLAT_CPLD_LPC_REG_TACHO5_OFFSET 0xe8
77 #define MLXPLAT_CPLD_LPC_REG_TACHO6_OFFSET 0xe9
78 #define MLXPLAT_CPLD_LPC_REG_TACHO7_OFFSET 0xeb
79 #define MLXPLAT_CPLD_LPC_REG_TACHO8_OFFSET 0xec
80 #define MLXPLAT_CPLD_LPC_REG_TACHO9_OFFSET 0xed
81 #define MLXPLAT_CPLD_LPC_REG_TACHO10_OFFSET 0xee
82 #define MLXPLAT_CPLD_LPC_REG_TACHO11_OFFSET 0xef
83 #define MLXPLAT_CPLD_LPC_REG_TACHO12_OFFSET 0xf0
84 #define MLXPLAT_CPLD_LPC_REG_FAN_CAP1_OFFSET 0xf5
85 #define MLXPLAT_CPLD_LPC_REG_FAN_CAP2_OFFSET 0xf6
86 #define MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET 0xf7
87 #define MLXPLAT_CPLD_LPC_REG_TACHO_SPEED_OFFSET 0xf8
88 #define MLXPLAT_CPLD_LPC_IO_RANGE 0x100
89 #define MLXPLAT_CPLD_LPC_I2C_CH1_OFF 0xdb
90 #define MLXPLAT_CPLD_LPC_I2C_CH2_OFF 0xda
91
92 #define MLXPLAT_CPLD_LPC_PIO_OFFSET 0x10000UL
93 #define MLXPLAT_CPLD_LPC_REG1 ((MLXPLAT_CPLD_LPC_REG_BASE_ADRR + \
94 MLXPLAT_CPLD_LPC_I2C_CH1_OFF) | \
95 MLXPLAT_CPLD_LPC_PIO_OFFSET)
96 #define MLXPLAT_CPLD_LPC_REG2 ((MLXPLAT_CPLD_LPC_REG_BASE_ADRR + \
97 MLXPLAT_CPLD_LPC_I2C_CH2_OFF) | \
98 MLXPLAT_CPLD_LPC_PIO_OFFSET)
99
100 /* Masks for aggregation, psu, pwr and fan event in CPLD related registers. */
101 #define MLXPLAT_CPLD_AGGR_ASIC_MASK_DEF 0x04
102 #define MLXPLAT_CPLD_AGGR_PSU_MASK_DEF 0x08
103 #define MLXPLAT_CPLD_AGGR_PWR_MASK_DEF 0x08
104 #define MLXPLAT_CPLD_AGGR_FAN_MASK_DEF 0x40
105 #define MLXPLAT_CPLD_AGGR_MASK_DEF (MLXPLAT_CPLD_AGGR_ASIC_MASK_DEF | \
106 MLXPLAT_CPLD_AGGR_PSU_MASK_DEF | \
107 MLXPLAT_CPLD_AGGR_FAN_MASK_DEF)
108 #define MLXPLAT_CPLD_AGGR_ASIC_MASK_NG 0x01
109 #define MLXPLAT_CPLD_AGGR_MASK_NG_DEF 0x04
110 #define MLXPLAT_CPLD_AGGR_MASK_COMEX BIT(0)
111 #define MLXPLAT_CPLD_LOW_AGGR_MASK_LOW 0xc1
112 #define MLXPLAT_CPLD_LOW_AGGR_MASK_I2C BIT(6)
113 #define MLXPLAT_CPLD_PSU_MASK GENMASK(1, 0)
114 #define MLXPLAT_CPLD_PWR_MASK GENMASK(1, 0)
115 #define MLXPLAT_CPLD_FAN_MASK GENMASK(3, 0)
116 #define MLXPLAT_CPLD_ASIC_MASK GENMASK(1, 0)
117 #define MLXPLAT_CPLD_FAN_NG_MASK GENMASK(5, 0)
118 #define MLXPLAT_CPLD_LED_LO_NIBBLE_MASK GENMASK(7, 4)
119 #define MLXPLAT_CPLD_LED_HI_NIBBLE_MASK GENMASK(3, 0)
120
121 /* Default I2C parent bus number */
122 #define MLXPLAT_CPLD_PHYS_ADAPTER_DEF_NR 1
123
124 /* Maximum number of possible physical buses equipped on system */
125 #define MLXPLAT_CPLD_MAX_PHYS_ADAPTER_NUM 16
126
127 /* Number of channels in group */
128 #define MLXPLAT_CPLD_GRP_CHNL_NUM 8
129
130 /* Start channel numbers */
131 #define MLXPLAT_CPLD_CH1 2
132 #define MLXPLAT_CPLD_CH2 10
133
134 /* Number of LPC attached MUX platform devices */
135 #define MLXPLAT_CPLD_LPC_MUX_DEVS 2
136
137 /* Hotplug devices adapter numbers */
138 #define MLXPLAT_CPLD_NR_NONE -1
139 #define MLXPLAT_CPLD_PSU_DEFAULT_NR 10
140 #define MLXPLAT_CPLD_PSU_MSNXXXX_NR 4
141 #define MLXPLAT_CPLD_FAN1_DEFAULT_NR 11
142 #define MLXPLAT_CPLD_FAN2_DEFAULT_NR 12
143 #define MLXPLAT_CPLD_FAN3_DEFAULT_NR 13
144 #define MLXPLAT_CPLD_FAN4_DEFAULT_NR 14
145
146 /* Masks and default values for watchdogs */
147 #define MLXPLAT_CPLD_WD1_CLEAR_MASK GENMASK(7, 1)
148 #define MLXPLAT_CPLD_WD2_CLEAR_MASK (GENMASK(7, 0) & ~BIT(1))
149
150 #define MLXPLAT_CPLD_WD_TYPE1_TO_MASK GENMASK(7, 4)
151 #define MLXPLAT_CPLD_WD_TYPE2_TO_MASK 0
152 #define MLXPLAT_CPLD_WD_RESET_ACT_MASK GENMASK(7, 1)
153 #define MLXPLAT_CPLD_WD_FAN_ACT_MASK (GENMASK(7, 0) & ~BIT(4))
154 #define MLXPLAT_CPLD_WD_COUNT_ACT_MASK (GENMASK(7, 0) & ~BIT(7))
155 #define MLXPLAT_CPLD_WD_DFLT_TIMEOUT 30
156 #define MLXPLAT_CPLD_WD_MAX_DEVS 2
157
158 /* mlxplat_priv - platform private data
159 * @pdev_i2c - i2c controller platform device
160 * @pdev_mux - array of mux platform devices
161 * @pdev_hotplug - hotplug platform devices
162 * @pdev_led - led platform devices
163 * @pdev_io_regs - register access platform devices
164 * @pdev_fan - FAN platform devices
165 * @pdev_wd - array of watchdog platform devices
166 * @regmap: device register map
167 */
168 struct mlxplat_priv {
169 struct platform_device *pdev_i2c;
170 struct platform_device *pdev_mux[MLXPLAT_CPLD_LPC_MUX_DEVS];
171 struct platform_device *pdev_hotplug;
172 struct platform_device *pdev_led;
173 struct platform_device *pdev_io_regs;
174 struct platform_device *pdev_fan;
175 struct platform_device *pdev_wd[MLXPLAT_CPLD_WD_MAX_DEVS];
176 void *regmap;
177 };
178
179 /* Regions for LPC I2C controller and LPC base register space */
180 static const struct resource mlxplat_lpc_resources[] = {
181 [0] = DEFINE_RES_NAMED(MLXPLAT_CPLD_LPC_I2C_BASE_ADRR,
182 MLXPLAT_CPLD_LPC_IO_RANGE,
183 "mlxplat_cpld_lpc_i2c_ctrl", IORESOURCE_IO),
184 [1] = DEFINE_RES_NAMED(MLXPLAT_CPLD_LPC_REG_BASE_ADRR,
185 MLXPLAT_CPLD_LPC_IO_RANGE,
186 "mlxplat_cpld_lpc_regs",
187 IORESOURCE_IO),
188 };
189
190 /* Platform next generation systems i2c data */
191 static struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_i2c_ng_data = {
192 .cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET,
193 .mask = MLXPLAT_CPLD_AGGR_MASK_COMEX,
194 .cell_low = MLXPLAT_CPLD_LPC_REG_AGGRCO_OFFSET,
195 .mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_I2C,
196 };
197
198 /* Platform default channels */
199 static const int mlxplat_default_channels[][MLXPLAT_CPLD_GRP_CHNL_NUM] = {
200 {
201 MLXPLAT_CPLD_CH1, MLXPLAT_CPLD_CH1 + 1, MLXPLAT_CPLD_CH1 + 2,
202 MLXPLAT_CPLD_CH1 + 3, MLXPLAT_CPLD_CH1 + 4, MLXPLAT_CPLD_CH1 +
203 5, MLXPLAT_CPLD_CH1 + 6, MLXPLAT_CPLD_CH1 + 7
204 },
205 {
206 MLXPLAT_CPLD_CH2, MLXPLAT_CPLD_CH2 + 1, MLXPLAT_CPLD_CH2 + 2,
207 MLXPLAT_CPLD_CH2 + 3, MLXPLAT_CPLD_CH2 + 4, MLXPLAT_CPLD_CH2 +
208 5, MLXPLAT_CPLD_CH2 + 6, MLXPLAT_CPLD_CH2 + 7
209 },
210 };
211
212 /* Platform channels for MSN21xx system family */
213 static const int mlxplat_msn21xx_channels[] = { 1, 2, 3, 4, 5, 6, 7, 8 };
214
215 /* Platform mux data */
216 static struct i2c_mux_reg_platform_data mlxplat_mux_data[] = {
217 {
218 .parent = 1,
219 .base_nr = MLXPLAT_CPLD_CH1,
220 .write_only = 1,
221 .reg = (void __iomem *)MLXPLAT_CPLD_LPC_REG1,
222 .reg_size = 1,
223 .idle_in_use = 1,
224 },
225 {
226 .parent = 1,
227 .base_nr = MLXPLAT_CPLD_CH2,
228 .write_only = 1,
229 .reg = (void __iomem *)MLXPLAT_CPLD_LPC_REG2,
230 .reg_size = 1,
231 .idle_in_use = 1,
232 },
233
234 };
235
236 /* Platform hotplug devices */
237 static struct i2c_board_info mlxplat_mlxcpld_psu[] = {
238 {
239 I2C_BOARD_INFO("24c02", 0x51),
240 },
241 {
242 I2C_BOARD_INFO("24c02", 0x50),
243 },
244 };
245
246 static struct i2c_board_info mlxplat_mlxcpld_ng_psu[] = {
247 {
248 I2C_BOARD_INFO("24c32", 0x51),
249 },
250 {
251 I2C_BOARD_INFO("24c32", 0x50),
252 },
253 };
254
255 static struct i2c_board_info mlxplat_mlxcpld_pwr[] = {
256 {
257 I2C_BOARD_INFO("dps460", 0x59),
258 },
259 {
260 I2C_BOARD_INFO("dps460", 0x58),
261 },
262 };
263
264 static struct i2c_board_info mlxplat_mlxcpld_fan[] = {
265 {
266 I2C_BOARD_INFO("24c32", 0x50),
267 },
268 {
269 I2C_BOARD_INFO("24c32", 0x50),
270 },
271 {
272 I2C_BOARD_INFO("24c32", 0x50),
273 },
274 {
275 I2C_BOARD_INFO("24c32", 0x50),
276 },
277 };
278
279 /* Platform hotplug default data */
280 static struct mlxreg_core_data mlxplat_mlxcpld_default_psu_items_data[] = {
281 {
282 .label = "psu1",
283 .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
284 .mask = BIT(0),
285 .hpdev.brdinfo = &mlxplat_mlxcpld_psu[0],
286 .hpdev.nr = MLXPLAT_CPLD_PSU_DEFAULT_NR,
287 },
288 {
289 .label = "psu2",
290 .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
291 .mask = BIT(1),
292 .hpdev.brdinfo = &mlxplat_mlxcpld_psu[1],
293 .hpdev.nr = MLXPLAT_CPLD_PSU_DEFAULT_NR,
294 },
295 };
296
297 static struct mlxreg_core_data mlxplat_mlxcpld_default_pwr_items_data[] = {
298 {
299 .label = "pwr1",
300 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
301 .mask = BIT(0),
302 .hpdev.brdinfo = &mlxplat_mlxcpld_pwr[0],
303 .hpdev.nr = MLXPLAT_CPLD_PSU_DEFAULT_NR,
304 },
305 {
306 .label = "pwr2",
307 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
308 .mask = BIT(1),
309 .hpdev.brdinfo = &mlxplat_mlxcpld_pwr[1],
310 .hpdev.nr = MLXPLAT_CPLD_PSU_DEFAULT_NR,
311 },
312 };
313
314 static struct mlxreg_core_data mlxplat_mlxcpld_default_fan_items_data[] = {
315 {
316 .label = "fan1",
317 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
318 .mask = BIT(0),
319 .hpdev.brdinfo = &mlxplat_mlxcpld_fan[0],
320 .hpdev.nr = MLXPLAT_CPLD_FAN1_DEFAULT_NR,
321 },
322 {
323 .label = "fan2",
324 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
325 .mask = BIT(1),
326 .hpdev.brdinfo = &mlxplat_mlxcpld_fan[1],
327 .hpdev.nr = MLXPLAT_CPLD_FAN2_DEFAULT_NR,
328 },
329 {
330 .label = "fan3",
331 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
332 .mask = BIT(2),
333 .hpdev.brdinfo = &mlxplat_mlxcpld_fan[2],
334 .hpdev.nr = MLXPLAT_CPLD_FAN3_DEFAULT_NR,
335 },
336 {
337 .label = "fan4",
338 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
339 .mask = BIT(3),
340 .hpdev.brdinfo = &mlxplat_mlxcpld_fan[3],
341 .hpdev.nr = MLXPLAT_CPLD_FAN4_DEFAULT_NR,
342 },
343 };
344
345 static struct mlxreg_core_data mlxplat_mlxcpld_default_asic_items_data[] = {
346 {
347 .label = "asic1",
348 .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
349 .mask = MLXPLAT_CPLD_ASIC_MASK,
350 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
351 },
352 };
353
354 static struct mlxreg_core_item mlxplat_mlxcpld_default_items[] = {
355 {
356 .data = mlxplat_mlxcpld_default_psu_items_data,
357 .aggr_mask = MLXPLAT_CPLD_AGGR_PSU_MASK_DEF,
358 .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
359 .mask = MLXPLAT_CPLD_PSU_MASK,
360 .count = ARRAY_SIZE(mlxplat_mlxcpld_psu),
361 .inversed = 1,
362 .health = false,
363 },
364 {
365 .data = mlxplat_mlxcpld_default_pwr_items_data,
366 .aggr_mask = MLXPLAT_CPLD_AGGR_PWR_MASK_DEF,
367 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
368 .mask = MLXPLAT_CPLD_PWR_MASK,
369 .count = ARRAY_SIZE(mlxplat_mlxcpld_pwr),
370 .inversed = 0,
371 .health = false,
372 },
373 {
374 .data = mlxplat_mlxcpld_default_fan_items_data,
375 .aggr_mask = MLXPLAT_CPLD_AGGR_FAN_MASK_DEF,
376 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
377 .mask = MLXPLAT_CPLD_FAN_MASK,
378 .count = ARRAY_SIZE(mlxplat_mlxcpld_fan),
379 .inversed = 1,
380 .health = false,
381 },
382 {
383 .data = mlxplat_mlxcpld_default_asic_items_data,
384 .aggr_mask = MLXPLAT_CPLD_AGGR_ASIC_MASK_DEF,
385 .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
386 .mask = MLXPLAT_CPLD_ASIC_MASK,
387 .count = ARRAY_SIZE(mlxplat_mlxcpld_default_asic_items_data),
388 .inversed = 0,
389 .health = true,
390 },
391 };
392
393 static
394 struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_default_data = {
395 .items = mlxplat_mlxcpld_default_items,
396 .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_items),
397 .cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET,
398 .mask = MLXPLAT_CPLD_AGGR_MASK_DEF,
399 .cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET,
400 .mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW,
401 };
402
403 static struct mlxreg_core_data mlxplat_mlxcpld_msn21xx_pwr_items_data[] = {
404 {
405 .label = "pwr1",
406 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
407 .mask = BIT(0),
408 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
409 },
410 {
411 .label = "pwr2",
412 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
413 .mask = BIT(1),
414 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
415 },
416 };
417
418 /* Platform hotplug MSN21xx system family data */
419 static struct mlxreg_core_item mlxplat_mlxcpld_msn21xx_items[] = {
420 {
421 .data = mlxplat_mlxcpld_msn21xx_pwr_items_data,
422 .aggr_mask = MLXPLAT_CPLD_AGGR_PWR_MASK_DEF,
423 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
424 .mask = MLXPLAT_CPLD_PWR_MASK,
425 .count = ARRAY_SIZE(mlxplat_mlxcpld_msn21xx_pwr_items_data),
426 .inversed = 0,
427 .health = false,
428 },
429 {
430 .data = mlxplat_mlxcpld_default_asic_items_data,
431 .aggr_mask = MLXPLAT_CPLD_AGGR_ASIC_MASK_DEF,
432 .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
433 .mask = MLXPLAT_CPLD_ASIC_MASK,
434 .count = ARRAY_SIZE(mlxplat_mlxcpld_default_asic_items_data),
435 .inversed = 0,
436 .health = true,
437 },
438 };
439
440 static
441 struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_msn21xx_data = {
442 .items = mlxplat_mlxcpld_msn21xx_items,
443 .counter = ARRAY_SIZE(mlxplat_mlxcpld_msn21xx_items),
444 .cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET,
445 .mask = MLXPLAT_CPLD_AGGR_MASK_DEF,
446 .cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET,
447 .mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW,
448 };
449
450 /* Platform hotplug msn274x system family data */
451 static struct mlxreg_core_data mlxplat_mlxcpld_msn274x_psu_items_data[] = {
452 {
453 .label = "psu1",
454 .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
455 .mask = BIT(0),
456 .hpdev.brdinfo = &mlxplat_mlxcpld_psu[0],
457 .hpdev.nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR,
458 },
459 {
460 .label = "psu2",
461 .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
462 .mask = BIT(1),
463 .hpdev.brdinfo = &mlxplat_mlxcpld_psu[1],
464 .hpdev.nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR,
465 },
466 };
467
468 static struct mlxreg_core_data mlxplat_mlxcpld_default_ng_pwr_items_data[] = {
469 {
470 .label = "pwr1",
471 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
472 .mask = BIT(0),
473 .hpdev.brdinfo = &mlxplat_mlxcpld_pwr[0],
474 .hpdev.nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR,
475 },
476 {
477 .label = "pwr2",
478 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
479 .mask = BIT(1),
480 .hpdev.brdinfo = &mlxplat_mlxcpld_pwr[1],
481 .hpdev.nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR,
482 },
483 };
484
485 static struct mlxreg_core_data mlxplat_mlxcpld_msn274x_fan_items_data[] = {
486 {
487 .label = "fan1",
488 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
489 .mask = BIT(0),
490 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
491 },
492 {
493 .label = "fan2",
494 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
495 .mask = BIT(1),
496 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
497 },
498 {
499 .label = "fan3",
500 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
501 .mask = BIT(2),
502 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
503 },
504 {
505 .label = "fan4",
506 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
507 .mask = BIT(3),
508 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
509 },
510 };
511
512 static struct mlxreg_core_item mlxplat_mlxcpld_msn274x_items[] = {
513 {
514 .data = mlxplat_mlxcpld_msn274x_psu_items_data,
515 .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
516 .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
517 .mask = MLXPLAT_CPLD_PSU_MASK,
518 .count = ARRAY_SIZE(mlxplat_mlxcpld_msn274x_psu_items_data),
519 .inversed = 1,
520 .health = false,
521 },
522 {
523 .data = mlxplat_mlxcpld_default_ng_pwr_items_data,
524 .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
525 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
526 .mask = MLXPLAT_CPLD_PWR_MASK,
527 .count = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_pwr_items_data),
528 .inversed = 0,
529 .health = false,
530 },
531 {
532 .data = mlxplat_mlxcpld_msn274x_fan_items_data,
533 .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
534 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
535 .mask = MLXPLAT_CPLD_FAN_MASK,
536 .count = ARRAY_SIZE(mlxplat_mlxcpld_msn274x_fan_items_data),
537 .inversed = 1,
538 .health = false,
539 },
540 {
541 .data = mlxplat_mlxcpld_default_asic_items_data,
542 .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
543 .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
544 .mask = MLXPLAT_CPLD_ASIC_MASK,
545 .count = ARRAY_SIZE(mlxplat_mlxcpld_default_asic_items_data),
546 .inversed = 0,
547 .health = true,
548 },
549 };
550
551 static
552 struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_msn274x_data = {
553 .items = mlxplat_mlxcpld_msn274x_items,
554 .counter = ARRAY_SIZE(mlxplat_mlxcpld_msn274x_items),
555 .cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET,
556 .mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
557 .cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET,
558 .mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW,
559 };
560
561 /* Platform hotplug MSN201x system family data */
562 static struct mlxreg_core_data mlxplat_mlxcpld_msn201x_pwr_items_data[] = {
563 {
564 .label = "pwr1",
565 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
566 .mask = BIT(0),
567 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
568 },
569 {
570 .label = "pwr2",
571 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
572 .mask = BIT(1),
573 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
574 },
575 };
576
577 static struct mlxreg_core_item mlxplat_mlxcpld_msn201x_items[] = {
578 {
579 .data = mlxplat_mlxcpld_msn201x_pwr_items_data,
580 .aggr_mask = MLXPLAT_CPLD_AGGR_PWR_MASK_DEF,
581 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
582 .mask = MLXPLAT_CPLD_PWR_MASK,
583 .count = ARRAY_SIZE(mlxplat_mlxcpld_msn201x_pwr_items_data),
584 .inversed = 0,
585 .health = false,
586 },
587 {
588 .data = mlxplat_mlxcpld_default_asic_items_data,
589 .aggr_mask = MLXPLAT_CPLD_AGGR_ASIC_MASK_DEF,
590 .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
591 .mask = MLXPLAT_CPLD_ASIC_MASK,
592 .count = ARRAY_SIZE(mlxplat_mlxcpld_default_asic_items_data),
593 .inversed = 0,
594 .health = true,
595 },
596 };
597
598 static
599 struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_msn201x_data = {
600 .items = mlxplat_mlxcpld_msn201x_items,
601 .counter = ARRAY_SIZE(mlxplat_mlxcpld_msn201x_items),
602 .cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET,
603 .mask = MLXPLAT_CPLD_AGGR_MASK_DEF,
604 .cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET,
605 .mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW,
606 };
607
608 /* Platform hotplug next generation system family data */
609 static struct mlxreg_core_data mlxplat_mlxcpld_default_ng_psu_items_data[] = {
610 {
611 .label = "psu1",
612 .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
613 .mask = BIT(0),
614 .hpdev.brdinfo = &mlxplat_mlxcpld_ng_psu[0],
615 .hpdev.nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR,
616 },
617 {
618 .label = "psu2",
619 .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
620 .mask = BIT(1),
621 .hpdev.brdinfo = &mlxplat_mlxcpld_ng_psu[1],
622 .hpdev.nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR,
623 },
624 };
625
626 static struct mlxreg_core_data mlxplat_mlxcpld_default_ng_fan_items_data[] = {
627 {
628 .label = "fan1",
629 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
630 .mask = BIT(0),
631 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
632 .bit = BIT(0),
633 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
634 },
635 {
636 .label = "fan2",
637 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
638 .mask = BIT(1),
639 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
640 .bit = BIT(1),
641 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
642 },
643 {
644 .label = "fan3",
645 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
646 .mask = BIT(2),
647 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
648 .bit = BIT(2),
649 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
650 },
651 {
652 .label = "fan4",
653 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
654 .mask = BIT(3),
655 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
656 .bit = BIT(3),
657 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
658 },
659 {
660 .label = "fan5",
661 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
662 .mask = BIT(4),
663 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
664 .bit = BIT(4),
665 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
666 },
667 {
668 .label = "fan6",
669 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
670 .mask = BIT(5),
671 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
672 .bit = BIT(5),
673 .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
674 },
675 };
676
677 static struct mlxreg_core_item mlxplat_mlxcpld_default_ng_items[] = {
678 {
679 .data = mlxplat_mlxcpld_default_ng_psu_items_data,
680 .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
681 .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
682 .mask = MLXPLAT_CPLD_PSU_MASK,
683 .count = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_psu_items_data),
684 .inversed = 1,
685 .health = false,
686 },
687 {
688 .data = mlxplat_mlxcpld_default_ng_pwr_items_data,
689 .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
690 .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
691 .mask = MLXPLAT_CPLD_PWR_MASK,
692 .count = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_pwr_items_data),
693 .inversed = 0,
694 .health = false,
695 },
696 {
697 .data = mlxplat_mlxcpld_default_ng_fan_items_data,
698 .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
699 .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
700 .mask = MLXPLAT_CPLD_FAN_NG_MASK,
701 .count = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_fan_items_data),
702 .inversed = 1,
703 .health = false,
704 },
705 {
706 .data = mlxplat_mlxcpld_default_asic_items_data,
707 .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
708 .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
709 .mask = MLXPLAT_CPLD_ASIC_MASK,
710 .count = ARRAY_SIZE(mlxplat_mlxcpld_default_asic_items_data),
711 .inversed = 0,
712 .health = true,
713 },
714 };
715
716 static
717 struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_default_ng_data = {
718 .items = mlxplat_mlxcpld_default_ng_items,
719 .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_items),
720 .cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET,
721 .mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF | MLXPLAT_CPLD_AGGR_MASK_COMEX,
722 .cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET,
723 .mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW,
724 };
725
726 /* Platform led default data */
727 static struct mlxreg_core_data mlxplat_mlxcpld_default_led_data[] = {
728 {
729 .label = "status:green",
730 .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
731 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
732 },
733 {
734 .label = "status:red",
735 .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
736 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK
737 },
738 {
739 .label = "psu:green",
740 .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
741 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
742 },
743 {
744 .label = "psu:red",
745 .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
746 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
747 },
748 {
749 .label = "fan1:green",
750 .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
751 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
752 },
753 {
754 .label = "fan1:red",
755 .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
756 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
757 },
758 {
759 .label = "fan2:green",
760 .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
761 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
762 },
763 {
764 .label = "fan2:red",
765 .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
766 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
767 },
768 {
769 .label = "fan3:green",
770 .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
771 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
772 },
773 {
774 .label = "fan3:red",
775 .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
776 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
777 },
778 {
779 .label = "fan4:green",
780 .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
781 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
782 },
783 {
784 .label = "fan4:red",
785 .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
786 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
787 },
788 };
789
790 static struct mlxreg_core_platform_data mlxplat_default_led_data = {
791 .data = mlxplat_mlxcpld_default_led_data,
792 .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_led_data),
793 };
794
795 /* Platform led MSN21xx system family data */
796 static struct mlxreg_core_data mlxplat_mlxcpld_msn21xx_led_data[] = {
797 {
798 .label = "status:green",
799 .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
800 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
801 },
802 {
803 .label = "status:red",
804 .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
805 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK
806 },
807 {
808 .label = "fan:green",
809 .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
810 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
811 },
812 {
813 .label = "fan:red",
814 .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
815 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
816 },
817 {
818 .label = "psu1:green",
819 .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
820 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
821 },
822 {
823 .label = "psu1:red",
824 .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
825 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
826 },
827 {
828 .label = "psu2:green",
829 .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
830 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
831 },
832 {
833 .label = "psu2:red",
834 .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
835 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
836 },
837 {
838 .label = "uid:blue",
839 .reg = MLXPLAT_CPLD_LPC_REG_LED5_OFFSET,
840 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
841 },
842 };
843
844 static struct mlxreg_core_platform_data mlxplat_msn21xx_led_data = {
845 .data = mlxplat_mlxcpld_msn21xx_led_data,
846 .counter = ARRAY_SIZE(mlxplat_mlxcpld_msn21xx_led_data),
847 };
848
849 /* Platform led for default data for 200GbE systems */
850 static struct mlxreg_core_data mlxplat_mlxcpld_default_ng_led_data[] = {
851 {
852 .label = "status:green",
853 .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
854 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
855 },
856 {
857 .label = "status:orange",
858 .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
859 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK
860 },
861 {
862 .label = "psu:green",
863 .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
864 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
865 },
866 {
867 .label = "psu:orange",
868 .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
869 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
870 },
871 {
872 .label = "fan1:green",
873 .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
874 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
875 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
876 .bit = BIT(0),
877 },
878 {
879 .label = "fan1:orange",
880 .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
881 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
882 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
883 .bit = BIT(0),
884 },
885 {
886 .label = "fan2:green",
887 .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
888 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
889 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
890 .bit = BIT(1),
891 },
892 {
893 .label = "fan2:orange",
894 .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
895 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
896 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
897 .bit = BIT(1),
898 },
899 {
900 .label = "fan3:green",
901 .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
902 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
903 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
904 .bit = BIT(2),
905 },
906 {
907 .label = "fan3:orange",
908 .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
909 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
910 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
911 .bit = BIT(2),
912 },
913 {
914 .label = "fan4:green",
915 .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
916 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
917 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
918 .bit = BIT(3),
919 },
920 {
921 .label = "fan4:orange",
922 .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
923 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
924 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
925 .bit = BIT(3),
926 },
927 {
928 .label = "fan5:green",
929 .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
930 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
931 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
932 .bit = BIT(4),
933 },
934 {
935 .label = "fan5:orange",
936 .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
937 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
938 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
939 .bit = BIT(4),
940 },
941 {
942 .label = "fan6:green",
943 .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
944 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
945 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
946 .bit = BIT(5),
947 },
948 {
949 .label = "fan6:orange",
950 .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
951 .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
952 .capability = MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET,
953 .bit = BIT(5),
954 },
955 {
956 .label = "uid:blue",
957 .reg = MLXPLAT_CPLD_LPC_REG_LED5_OFFSET,
958 .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
959 },
960 };
961
962 static struct mlxreg_core_platform_data mlxplat_default_ng_led_data = {
963 .data = mlxplat_mlxcpld_default_ng_led_data,
964 .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_led_data),
965 };
966
967 /* Platform register access default */
968 static struct mlxreg_core_data mlxplat_mlxcpld_default_regs_io_data[] = {
969 {
970 .label = "cpld1_version",
971 .reg = MLXPLAT_CPLD_LPC_REG_CPLD1_VER_OFFSET,
972 .bit = GENMASK(7, 0),
973 .mode = 0444,
974 },
975 {
976 .label = "cpld2_version",
977 .reg = MLXPLAT_CPLD_LPC_REG_CPLD2_VER_OFFSET,
978 .bit = GENMASK(7, 0),
979 .mode = 0444,
980 },
981 {
982 .label = "reset_long_pb",
983 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
984 .mask = GENMASK(7, 0) & ~BIT(0),
985 .mode = 0444,
986 },
987 {
988 .label = "reset_short_pb",
989 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
990 .mask = GENMASK(7, 0) & ~BIT(1),
991 .mode = 0444,
992 },
993 {
994 .label = "reset_aux_pwr_or_ref",
995 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
996 .mask = GENMASK(7, 0) & ~BIT(2),
997 .mode = 0444,
998 },
999 {
1000 .label = "reset_main_pwr_fail",
1001 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1002 .mask = GENMASK(7, 0) & ~BIT(3),
1003 .mode = 0444,
1004 },
1005 {
1006 .label = "reset_sw_reset",
1007 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1008 .mask = GENMASK(7, 0) & ~BIT(4),
1009 .mode = 0444,
1010 },
1011 {
1012 .label = "reset_fw_reset",
1013 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1014 .mask = GENMASK(7, 0) & ~BIT(5),
1015 .mode = 0444,
1016 },
1017 {
1018 .label = "reset_hotswap_or_wd",
1019 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1020 .mask = GENMASK(7, 0) & ~BIT(6),
1021 .mode = 0444,
1022 },
1023 {
1024 .label = "reset_asic_thermal",
1025 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1026 .mask = GENMASK(7, 0) & ~BIT(7),
1027 .mode = 0444,
1028 },
1029 {
1030 .label = "psu1_on",
1031 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1032 .mask = GENMASK(7, 0) & ~BIT(0),
1033 .mode = 0200,
1034 },
1035 {
1036 .label = "psu2_on",
1037 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1038 .mask = GENMASK(7, 0) & ~BIT(1),
1039 .mode = 0200,
1040 },
1041 {
1042 .label = "pwr_cycle",
1043 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1044 .mask = GENMASK(7, 0) & ~BIT(2),
1045 .mode = 0200,
1046 },
1047 {
1048 .label = "pwr_down",
1049 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1050 .mask = GENMASK(7, 0) & ~BIT(3),
1051 .mode = 0200,
1052 },
1053 {
1054 .label = "select_iio",
1055 .reg = MLXPLAT_CPLD_LPC_REG_GP2_OFFSET,
1056 .mask = GENMASK(7, 0) & ~BIT(6),
1057 .mode = 0644,
1058 },
1059 {
1060 .label = "asic_health",
1061 .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
1062 .mask = MLXPLAT_CPLD_ASIC_MASK,
1063 .bit = 1,
1064 .mode = 0444,
1065 },
1066 };
1067
1068 static struct mlxreg_core_platform_data mlxplat_default_regs_io_data = {
1069 .data = mlxplat_mlxcpld_default_regs_io_data,
1070 .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_regs_io_data),
1071 };
1072
1073 /* Platform register access MSN21xx, MSN201x, MSN274x systems families data */
1074 static struct mlxreg_core_data mlxplat_mlxcpld_msn21xx_regs_io_data[] = {
1075 {
1076 .label = "cpld1_version",
1077 .reg = MLXPLAT_CPLD_LPC_REG_CPLD1_VER_OFFSET,
1078 .bit = GENMASK(7, 0),
1079 .mode = 0444,
1080 },
1081 {
1082 .label = "cpld2_version",
1083 .reg = MLXPLAT_CPLD_LPC_REG_CPLD2_VER_OFFSET,
1084 .bit = GENMASK(7, 0),
1085 .mode = 0444,
1086 },
1087 {
1088 .label = "reset_long_pb",
1089 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1090 .mask = GENMASK(7, 0) & ~BIT(0),
1091 .mode = 0444,
1092 },
1093 {
1094 .label = "reset_short_pb",
1095 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1096 .mask = GENMASK(7, 0) & ~BIT(1),
1097 .mode = 0444,
1098 },
1099 {
1100 .label = "reset_aux_pwr_or_ref",
1101 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1102 .mask = GENMASK(7, 0) & ~BIT(2),
1103 .mode = 0444,
1104 },
1105 {
1106 .label = "reset_sw_reset",
1107 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1108 .mask = GENMASK(7, 0) & ~BIT(3),
1109 .mode = 0444,
1110 },
1111 {
1112 .label = "reset_main_pwr_fail",
1113 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1114 .mask = GENMASK(7, 0) & ~BIT(4),
1115 .mode = 0444,
1116 },
1117 {
1118 .label = "reset_asic_thermal",
1119 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1120 .mask = GENMASK(7, 0) & ~BIT(5),
1121 .mode = 0444,
1122 },
1123 {
1124 .label = "reset_hotswap_or_halt",
1125 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1126 .mask = GENMASK(7, 0) & ~BIT(6),
1127 .mode = 0444,
1128 },
1129 {
1130 .label = "reset_sff_wd",
1131 .reg = MLXPLAT_CPLD_LPC_REG_RST_CAUSE1_OFFSET,
1132 .mask = GENMASK(7, 0) & ~BIT(6),
1133 .mode = 0444,
1134 },
1135 {
1136 .label = "psu1_on",
1137 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1138 .mask = GENMASK(7, 0) & ~BIT(0),
1139 .mode = 0200,
1140 },
1141 {
1142 .label = "psu2_on",
1143 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1144 .mask = GENMASK(7, 0) & ~BIT(1),
1145 .mode = 0200,
1146 },
1147 {
1148 .label = "pwr_cycle",
1149 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1150 .mask = GENMASK(7, 0) & ~BIT(2),
1151 .mode = 0200,
1152 },
1153 {
1154 .label = "pwr_down",
1155 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1156 .mask = GENMASK(7, 0) & ~BIT(3),
1157 .mode = 0200,
1158 },
1159 {
1160 .label = "asic_health",
1161 .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
1162 .mask = MLXPLAT_CPLD_ASIC_MASK,
1163 .bit = 1,
1164 .mode = 0444,
1165 },
1166 };
1167
1168 static struct mlxreg_core_platform_data mlxplat_msn21xx_regs_io_data = {
1169 .data = mlxplat_mlxcpld_msn21xx_regs_io_data,
1170 .counter = ARRAY_SIZE(mlxplat_mlxcpld_msn21xx_regs_io_data),
1171 };
1172
1173 /* Platform register access for next generation systems families data */
1174 static struct mlxreg_core_data mlxplat_mlxcpld_default_ng_regs_io_data[] = {
1175 {
1176 .label = "cpld1_version",
1177 .reg = MLXPLAT_CPLD_LPC_REG_CPLD1_VER_OFFSET,
1178 .bit = GENMASK(7, 0),
1179 .mode = 0444,
1180 },
1181 {
1182 .label = "cpld2_version",
1183 .reg = MLXPLAT_CPLD_LPC_REG_CPLD2_VER_OFFSET,
1184 .bit = GENMASK(7, 0),
1185 .mode = 0444,
1186 },
1187 {
1188 .label = "cpld3_version",
1189 .reg = MLXPLAT_CPLD_LPC_REG_CPLD3_VER_OFFSET,
1190 .bit = GENMASK(7, 0),
1191 .mode = 0444,
1192 },
1193 {
1194 .label = "cpld4_version",
1195 .reg = MLXPLAT_CPLD_LPC_REG_CPLD4_VER_OFFSET,
1196 .bit = GENMASK(7, 0),
1197 .mode = 0444,
1198 },
1199 {
1200 .label = "reset_long_pb",
1201 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1202 .mask = GENMASK(7, 0) & ~BIT(0),
1203 .mode = 0444,
1204 },
1205 {
1206 .label = "reset_short_pb",
1207 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1208 .mask = GENMASK(7, 0) & ~BIT(1),
1209 .mode = 0444,
1210 },
1211 {
1212 .label = "reset_aux_pwr_or_ref",
1213 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1214 .mask = GENMASK(7, 0) & ~BIT(2),
1215 .mode = 0444,
1216 },
1217 {
1218 .label = "reset_from_comex",
1219 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1220 .mask = GENMASK(7, 0) & ~BIT(4),
1221 .mode = 0444,
1222 },
1223 {
1224 .label = "reset_from_asic",
1225 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1226 .mask = GENMASK(7, 0) & ~BIT(5),
1227 .mode = 0444,
1228 },
1229 {
1230 .label = "reset_swb_wd",
1231 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1232 .mask = GENMASK(7, 0) & ~BIT(6),
1233 .mode = 0444,
1234 },
1235 {
1236 .label = "reset_asic_thermal",
1237 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1238 .mask = GENMASK(7, 0) & ~BIT(7),
1239 .mode = 0444,
1240 },
1241 {
1242 .label = "reset_comex_pwr_fail",
1243 .reg = MLXPLAT_CPLD_LPC_REG_RST_CAUSE1_OFFSET,
1244 .mask = GENMASK(7, 0) & ~BIT(3),
1245 .mode = 0444,
1246 },
1247 {
1248 .label = "reset_comex_wd",
1249 .reg = MLXPLAT_CPLD_LPC_REG_RST_CAUSE1_OFFSET,
1250 .mask = GENMASK(7, 0) & ~BIT(6),
1251 .mode = 0444,
1252 },
1253 {
1254 .label = "reset_voltmon_upgrade_fail",
1255 .reg = MLXPLAT_CPLD_LPC_REG_RST_CAUSE2_OFFSET,
1256 .mask = GENMASK(7, 0) & ~BIT(0),
1257 .mode = 0444,
1258 },
1259 {
1260 .label = "reset_system",
1261 .reg = MLXPLAT_CPLD_LPC_REG_RST_CAUSE2_OFFSET,
1262 .mask = GENMASK(7, 0) & ~BIT(1),
1263 .mode = 0444,
1264 },
1265 {
1266 .label = "reset_comex_thermal",
1267 .reg = MLXPLAT_CPLD_LPC_REG_RST_CAUSE2_OFFSET,
1268 .mask = GENMASK(7, 0) & ~BIT(3),
1269 .mode = 0444,
1270 },
1271 {
1272 .label = "reset_reload_bios",
1273 .reg = MLXPLAT_CPLD_LPC_REG_RST_CAUSE2_OFFSET,
1274 .mask = GENMASK(7, 0) & ~BIT(5),
1275 .mode = 0444,
1276 },
1277 {
1278 .label = "psu1_on",
1279 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1280 .mask = GENMASK(7, 0) & ~BIT(0),
1281 .mode = 0200,
1282 },
1283 {
1284 .label = "psu2_on",
1285 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1286 .mask = GENMASK(7, 0) & ~BIT(1),
1287 .mode = 0200,
1288 },
1289 {
1290 .label = "pwr_cycle",
1291 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1292 .mask = GENMASK(7, 0) & ~BIT(2),
1293 .mode = 0200,
1294 },
1295 {
1296 .label = "pwr_down",
1297 .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
1298 .mask = GENMASK(7, 0) & ~BIT(3),
1299 .mode = 0200,
1300 },
1301 {
1302 .label = "jtag_enable",
1303 .reg = MLXPLAT_CPLD_LPC_REG_GP2_OFFSET,
1304 .mask = GENMASK(7, 0) & ~BIT(4),
1305 .mode = 0644,
1306 },
1307 {
1308 .label = "asic_health",
1309 .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
1310 .mask = MLXPLAT_CPLD_ASIC_MASK,
1311 .bit = 1,
1312 .mode = 0444,
1313 },
1314 {
1315 .label = "fan_dir",
1316 .reg = MLXPLAT_CPLD_LPC_REG_FAN_DIRECTION,
1317 .bit = GENMASK(7, 0),
1318 .mode = 0444,
1319 },
1320 };
1321
1322 static struct mlxreg_core_platform_data mlxplat_default_ng_regs_io_data = {
1323 .data = mlxplat_mlxcpld_default_ng_regs_io_data,
1324 .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_regs_io_data),
1325 };
1326
1327 /* Platform FAN default */
1328 static struct mlxreg_core_data mlxplat_mlxcpld_default_fan_data[] = {
1329 {
1330 .label = "pwm1",
1331 .reg = MLXPLAT_CPLD_LPC_REG_PWM1_OFFSET,
1332 },
1333 {
1334 .label = "tacho1",
1335 .reg = MLXPLAT_CPLD_LPC_REG_TACHO1_OFFSET,
1336 .mask = GENMASK(7, 0),
1337 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP1_OFFSET,
1338 .bit = BIT(0),
1339 },
1340 {
1341 .label = "tacho2",
1342 .reg = MLXPLAT_CPLD_LPC_REG_TACHO2_OFFSET,
1343 .mask = GENMASK(7, 0),
1344 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP1_OFFSET,
1345 .bit = BIT(1),
1346 },
1347 {
1348 .label = "tacho3",
1349 .reg = MLXPLAT_CPLD_LPC_REG_TACHO3_OFFSET,
1350 .mask = GENMASK(7, 0),
1351 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP1_OFFSET,
1352 .bit = BIT(2),
1353 },
1354 {
1355 .label = "tacho4",
1356 .reg = MLXPLAT_CPLD_LPC_REG_TACHO4_OFFSET,
1357 .mask = GENMASK(7, 0),
1358 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP1_OFFSET,
1359 .bit = BIT(3),
1360 },
1361 {
1362 .label = "tacho5",
1363 .reg = MLXPLAT_CPLD_LPC_REG_TACHO5_OFFSET,
1364 .mask = GENMASK(7, 0),
1365 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP1_OFFSET,
1366 .bit = BIT(4),
1367 },
1368 {
1369 .label = "tacho6",
1370 .reg = MLXPLAT_CPLD_LPC_REG_TACHO6_OFFSET,
1371 .mask = GENMASK(7, 0),
1372 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP1_OFFSET,
1373 .bit = BIT(5),
1374 },
1375 {
1376 .label = "tacho7",
1377 .reg = MLXPLAT_CPLD_LPC_REG_TACHO7_OFFSET,
1378 .mask = GENMASK(7, 0),
1379 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP1_OFFSET,
1380 .bit = BIT(6),
1381 },
1382 {
1383 .label = "tacho8",
1384 .reg = MLXPLAT_CPLD_LPC_REG_TACHO8_OFFSET,
1385 .mask = GENMASK(7, 0),
1386 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP1_OFFSET,
1387 .bit = BIT(7),
1388 },
1389 {
1390 .label = "tacho9",
1391 .reg = MLXPLAT_CPLD_LPC_REG_TACHO9_OFFSET,
1392 .mask = GENMASK(7, 0),
1393 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP2_OFFSET,
1394 .bit = BIT(0),
1395 },
1396 {
1397 .label = "tacho10",
1398 .reg = MLXPLAT_CPLD_LPC_REG_TACHO10_OFFSET,
1399 .mask = GENMASK(7, 0),
1400 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP2_OFFSET,
1401 .bit = BIT(1),
1402 },
1403 {
1404 .label = "tacho11",
1405 .reg = MLXPLAT_CPLD_LPC_REG_TACHO11_OFFSET,
1406 .mask = GENMASK(7, 0),
1407 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP2_OFFSET,
1408 .bit = BIT(2),
1409 },
1410 {
1411 .label = "tacho12",
1412 .reg = MLXPLAT_CPLD_LPC_REG_TACHO12_OFFSET,
1413 .mask = GENMASK(7, 0),
1414 .capability = MLXPLAT_CPLD_LPC_REG_FAN_CAP2_OFFSET,
1415 .bit = BIT(3),
1416 },
1417 {
1418 .label = "conf",
1419 .capability = MLXPLAT_CPLD_LPC_REG_TACHO_SPEED_OFFSET,
1420 },
1421 };
1422
1423 static struct mlxreg_core_platform_data mlxplat_default_fan_data = {
1424 .data = mlxplat_mlxcpld_default_fan_data,
1425 .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_fan_data),
1426 };
1427
1428 /* Watchdog type1: hardware implementation version1
1429 * (MSN2700, MSN2410, MSN2740, MSN2100 and MSN2140 systems).
1430 */
1431 static struct mlxreg_core_data mlxplat_mlxcpld_wd_main_regs_type1[] = {
1432 {
1433 .label = "action",
1434 .reg = MLXPLAT_CPLD_LPC_REG_WD1_ACT_OFFSET,
1435 .mask = MLXPLAT_CPLD_WD_RESET_ACT_MASK,
1436 .bit = 0,
1437 },
1438 {
1439 .label = "timeout",
1440 .reg = MLXPLAT_CPLD_LPC_REG_WD1_TMR_OFFSET,
1441 .mask = MLXPLAT_CPLD_WD_TYPE1_TO_MASK,
1442 .health_cntr = MLXPLAT_CPLD_WD_DFLT_TIMEOUT,
1443 },
1444 {
1445 .label = "ping",
1446 .reg = MLXPLAT_CPLD_LPC_REG_WD_CLEAR_OFFSET,
1447 .mask = MLXPLAT_CPLD_WD1_CLEAR_MASK,
1448 .bit = 0,
1449 },
1450 {
1451 .label = "reset",
1452 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1453 .mask = GENMASK(7, 0) & ~BIT(6),
1454 .bit = 6,
1455 },
1456 };
1457
1458 static struct mlxreg_core_data mlxplat_mlxcpld_wd_aux_regs_type1[] = {
1459 {
1460 .label = "action",
1461 .reg = MLXPLAT_CPLD_LPC_REG_WD2_ACT_OFFSET,
1462 .mask = MLXPLAT_CPLD_WD_FAN_ACT_MASK,
1463 .bit = 4,
1464 },
1465 {
1466 .label = "timeout",
1467 .reg = MLXPLAT_CPLD_LPC_REG_WD2_TMR_OFFSET,
1468 .mask = MLXPLAT_CPLD_WD_TYPE1_TO_MASK,
1469 .health_cntr = MLXPLAT_CPLD_WD_DFLT_TIMEOUT,
1470 },
1471 {
1472 .label = "ping",
1473 .reg = MLXPLAT_CPLD_LPC_REG_WD_CLEAR_OFFSET,
1474 .mask = MLXPLAT_CPLD_WD1_CLEAR_MASK,
1475 .bit = 1,
1476 },
1477 };
1478
1479 static struct mlxreg_core_platform_data mlxplat_mlxcpld_wd_set_type1[] = {
1480 {
1481 .data = mlxplat_mlxcpld_wd_main_regs_type1,
1482 .counter = ARRAY_SIZE(mlxplat_mlxcpld_wd_main_regs_type1),
1483 .version = MLX_WDT_TYPE1,
1484 .identity = "mlx-wdt-main",
1485 },
1486 {
1487 .data = mlxplat_mlxcpld_wd_aux_regs_type1,
1488 .counter = ARRAY_SIZE(mlxplat_mlxcpld_wd_aux_regs_type1),
1489 .version = MLX_WDT_TYPE1,
1490 .identity = "mlx-wdt-aux",
1491 },
1492 };
1493
1494 /* Watchdog type2: hardware implementation version 2
1495 * (all systems except (MSN2700, MSN2410, MSN2740, MSN2100 and MSN2140).
1496 */
1497 static struct mlxreg_core_data mlxplat_mlxcpld_wd_main_regs_type2[] = {
1498 {
1499 .label = "action",
1500 .reg = MLXPLAT_CPLD_LPC_REG_WD2_ACT_OFFSET,
1501 .mask = MLXPLAT_CPLD_WD_RESET_ACT_MASK,
1502 .bit = 0,
1503 },
1504 {
1505 .label = "timeout",
1506 .reg = MLXPLAT_CPLD_LPC_REG_WD2_TMR_OFFSET,
1507 .mask = MLXPLAT_CPLD_WD_TYPE2_TO_MASK,
1508 .health_cntr = MLXPLAT_CPLD_WD_DFLT_TIMEOUT,
1509 },
1510 {
1511 .label = "timeleft",
1512 .reg = MLXPLAT_CPLD_LPC_REG_WD2_TLEFT_OFFSET,
1513 .mask = MLXPLAT_CPLD_WD_TYPE2_TO_MASK,
1514 },
1515 {
1516 .label = "ping",
1517 .reg = MLXPLAT_CPLD_LPC_REG_WD2_ACT_OFFSET,
1518 .mask = MLXPLAT_CPLD_WD_RESET_ACT_MASK,
1519 .bit = 0,
1520 },
1521 {
1522 .label = "reset",
1523 .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
1524 .mask = GENMASK(7, 0) & ~BIT(6),
1525 .bit = 6,
1526 },
1527 };
1528
1529 static struct mlxreg_core_data mlxplat_mlxcpld_wd_aux_regs_type2[] = {
1530 {
1531 .label = "action",
1532 .reg = MLXPLAT_CPLD_LPC_REG_WD3_ACT_OFFSET,
1533 .mask = MLXPLAT_CPLD_WD_FAN_ACT_MASK,
1534 .bit = 4,
1535 },
1536 {
1537 .label = "timeout",
1538 .reg = MLXPLAT_CPLD_LPC_REG_WD3_TMR_OFFSET,
1539 .mask = MLXPLAT_CPLD_WD_TYPE2_TO_MASK,
1540 .health_cntr = MLXPLAT_CPLD_WD_DFLT_TIMEOUT,
1541 },
1542 {
1543 .label = "timeleft",
1544 .reg = MLXPLAT_CPLD_LPC_REG_WD3_TLEFT_OFFSET,
1545 .mask = MLXPLAT_CPLD_WD_TYPE2_TO_MASK,
1546 },
1547 {
1548 .label = "ping",
1549 .reg = MLXPLAT_CPLD_LPC_REG_WD3_ACT_OFFSET,
1550 .mask = MLXPLAT_CPLD_WD_FAN_ACT_MASK,
1551 .bit = 4,
1552 },
1553 };
1554
1555 static struct mlxreg_core_platform_data mlxplat_mlxcpld_wd_set_type2[] = {
1556 {
1557 .data = mlxplat_mlxcpld_wd_main_regs_type2,
1558 .counter = ARRAY_SIZE(mlxplat_mlxcpld_wd_main_regs_type2),
1559 .version = MLX_WDT_TYPE2,
1560 .identity = "mlx-wdt-main",
1561 },
1562 {
1563 .data = mlxplat_mlxcpld_wd_aux_regs_type2,
1564 .counter = ARRAY_SIZE(mlxplat_mlxcpld_wd_aux_regs_type2),
1565 .version = MLX_WDT_TYPE2,
1566 .identity = "mlx-wdt-aux",
1567 },
1568 };
1569
mlxplat_mlxcpld_writeable_reg(struct device * dev,unsigned int reg)1570 static bool mlxplat_mlxcpld_writeable_reg(struct device *dev, unsigned int reg)
1571 {
1572 switch (reg) {
1573 case MLXPLAT_CPLD_LPC_REG_LED1_OFFSET:
1574 case MLXPLAT_CPLD_LPC_REG_LED2_OFFSET:
1575 case MLXPLAT_CPLD_LPC_REG_LED3_OFFSET:
1576 case MLXPLAT_CPLD_LPC_REG_LED4_OFFSET:
1577 case MLXPLAT_CPLD_LPC_REG_LED5_OFFSET:
1578 case MLXPLAT_CPLD_LPC_REG_GP1_OFFSET:
1579 case MLXPLAT_CPLD_LPC_REG_WP1_OFFSET:
1580 case MLXPLAT_CPLD_LPC_REG_GP2_OFFSET:
1581 case MLXPLAT_CPLD_LPC_REG_WP2_OFFSET:
1582 case MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET:
1583 case MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET:
1584 case MLXPLAT_CPLD_LPC_REG_AGGRCO_MASK_OFFSET:
1585 case MLXPLAT_CPLD_LPC_REG_ASIC_EVENT_OFFSET:
1586 case MLXPLAT_CPLD_LPC_REG_ASIC_MASK_OFFSET:
1587 case MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET:
1588 case MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET:
1589 case MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET:
1590 case MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET:
1591 case MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET:
1592 case MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET:
1593 case MLXPLAT_CPLD_LPC_REG_WD_CLEAR_OFFSET:
1594 case MLXPLAT_CPLD_LPC_REG_WD_CLEAR_WP_OFFSET:
1595 case MLXPLAT_CPLD_LPC_REG_WD1_TMR_OFFSET:
1596 case MLXPLAT_CPLD_LPC_REG_WD1_ACT_OFFSET:
1597 case MLXPLAT_CPLD_LPC_REG_WD2_TMR_OFFSET:
1598 case MLXPLAT_CPLD_LPC_REG_WD2_ACT_OFFSET:
1599 case MLXPLAT_CPLD_LPC_REG_WD3_TMR_OFFSET:
1600 case MLXPLAT_CPLD_LPC_REG_WD3_ACT_OFFSET:
1601 case MLXPLAT_CPLD_LPC_REG_PWM1_OFFSET:
1602 case MLXPLAT_CPLD_LPC_REG_PWM_CONTROL_OFFSET:
1603 return true;
1604 }
1605 return false;
1606 }
1607
mlxplat_mlxcpld_readable_reg(struct device * dev,unsigned int reg)1608 static bool mlxplat_mlxcpld_readable_reg(struct device *dev, unsigned int reg)
1609 {
1610 switch (reg) {
1611 case MLXPLAT_CPLD_LPC_REG_CPLD1_VER_OFFSET:
1612 case MLXPLAT_CPLD_LPC_REG_CPLD2_VER_OFFSET:
1613 case MLXPLAT_CPLD_LPC_REG_CPLD3_VER_OFFSET:
1614 case MLXPLAT_CPLD_LPC_REG_CPLD4_VER_OFFSET:
1615 case MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET:
1616 case MLXPLAT_CPLD_LPC_REG_RST_CAUSE1_OFFSET:
1617 case MLXPLAT_CPLD_LPC_REG_RST_CAUSE2_OFFSET:
1618 case MLXPLAT_CPLD_LPC_REG_LED1_OFFSET:
1619 case MLXPLAT_CPLD_LPC_REG_LED2_OFFSET:
1620 case MLXPLAT_CPLD_LPC_REG_LED3_OFFSET:
1621 case MLXPLAT_CPLD_LPC_REG_LED4_OFFSET:
1622 case MLXPLAT_CPLD_LPC_REG_LED5_OFFSET:
1623 case MLXPLAT_CPLD_LPC_REG_FAN_DIRECTION:
1624 case MLXPLAT_CPLD_LPC_REG_GP1_OFFSET:
1625 case MLXPLAT_CPLD_LPC_REG_WP1_OFFSET:
1626 case MLXPLAT_CPLD_LPC_REG_GP2_OFFSET:
1627 case MLXPLAT_CPLD_LPC_REG_WP2_OFFSET:
1628 case MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET:
1629 case MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET:
1630 case MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET:
1631 case MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET:
1632 case MLXPLAT_CPLD_LPC_REG_AGGRCO_OFFSET:
1633 case MLXPLAT_CPLD_LPC_REG_AGGRCO_MASK_OFFSET:
1634 case MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET:
1635 case MLXPLAT_CPLD_LPC_REG_ASIC_EVENT_OFFSET:
1636 case MLXPLAT_CPLD_LPC_REG_ASIC_MASK_OFFSET:
1637 case MLXPLAT_CPLD_LPC_REG_PSU_OFFSET:
1638 case MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET:
1639 case MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET:
1640 case MLXPLAT_CPLD_LPC_REG_PWR_OFFSET:
1641 case MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET:
1642 case MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET:
1643 case MLXPLAT_CPLD_LPC_REG_FAN_OFFSET:
1644 case MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET:
1645 case MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET:
1646 case MLXPLAT_CPLD_LPC_REG_WD_CLEAR_OFFSET:
1647 case MLXPLAT_CPLD_LPC_REG_WD_CLEAR_WP_OFFSET:
1648 case MLXPLAT_CPLD_LPC_REG_WD1_TMR_OFFSET:
1649 case MLXPLAT_CPLD_LPC_REG_WD1_ACT_OFFSET:
1650 case MLXPLAT_CPLD_LPC_REG_WD2_TMR_OFFSET:
1651 case MLXPLAT_CPLD_LPC_REG_WD2_TLEFT_OFFSET:
1652 case MLXPLAT_CPLD_LPC_REG_WD2_ACT_OFFSET:
1653 case MLXPLAT_CPLD_LPC_REG_WD3_TMR_OFFSET:
1654 case MLXPLAT_CPLD_LPC_REG_WD3_TLEFT_OFFSET:
1655 case MLXPLAT_CPLD_LPC_REG_WD3_ACT_OFFSET:
1656 case MLXPLAT_CPLD_LPC_REG_PWM1_OFFSET:
1657 case MLXPLAT_CPLD_LPC_REG_TACHO1_OFFSET:
1658 case MLXPLAT_CPLD_LPC_REG_TACHO2_OFFSET:
1659 case MLXPLAT_CPLD_LPC_REG_TACHO3_OFFSET:
1660 case MLXPLAT_CPLD_LPC_REG_TACHO4_OFFSET:
1661 case MLXPLAT_CPLD_LPC_REG_TACHO5_OFFSET:
1662 case MLXPLAT_CPLD_LPC_REG_TACHO6_OFFSET:
1663 case MLXPLAT_CPLD_LPC_REG_TACHO7_OFFSET:
1664 case MLXPLAT_CPLD_LPC_REG_TACHO8_OFFSET:
1665 case MLXPLAT_CPLD_LPC_REG_TACHO9_OFFSET:
1666 case MLXPLAT_CPLD_LPC_REG_TACHO10_OFFSET:
1667 case MLXPLAT_CPLD_LPC_REG_TACHO11_OFFSET:
1668 case MLXPLAT_CPLD_LPC_REG_TACHO12_OFFSET:
1669 case MLXPLAT_CPLD_LPC_REG_PWM_CONTROL_OFFSET:
1670 case MLXPLAT_CPLD_LPC_REG_FAN_CAP1_OFFSET:
1671 case MLXPLAT_CPLD_LPC_REG_FAN_CAP2_OFFSET:
1672 case MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET:
1673 case MLXPLAT_CPLD_LPC_REG_TACHO_SPEED_OFFSET:
1674 return true;
1675 }
1676 return false;
1677 }
1678
mlxplat_mlxcpld_volatile_reg(struct device * dev,unsigned int reg)1679 static bool mlxplat_mlxcpld_volatile_reg(struct device *dev, unsigned int reg)
1680 {
1681 switch (reg) {
1682 case MLXPLAT_CPLD_LPC_REG_CPLD1_VER_OFFSET:
1683 case MLXPLAT_CPLD_LPC_REG_CPLD2_VER_OFFSET:
1684 case MLXPLAT_CPLD_LPC_REG_CPLD3_VER_OFFSET:
1685 case MLXPLAT_CPLD_LPC_REG_CPLD4_VER_OFFSET:
1686 case MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET:
1687 case MLXPLAT_CPLD_LPC_REG_RST_CAUSE1_OFFSET:
1688 case MLXPLAT_CPLD_LPC_REG_RST_CAUSE2_OFFSET:
1689 case MLXPLAT_CPLD_LPC_REG_LED1_OFFSET:
1690 case MLXPLAT_CPLD_LPC_REG_LED2_OFFSET:
1691 case MLXPLAT_CPLD_LPC_REG_LED3_OFFSET:
1692 case MLXPLAT_CPLD_LPC_REG_LED4_OFFSET:
1693 case MLXPLAT_CPLD_LPC_REG_LED5_OFFSET:
1694 case MLXPLAT_CPLD_LPC_REG_FAN_DIRECTION:
1695 case MLXPLAT_CPLD_LPC_REG_GP1_OFFSET:
1696 case MLXPLAT_CPLD_LPC_REG_GP2_OFFSET:
1697 case MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET:
1698 case MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET:
1699 case MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET:
1700 case MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET:
1701 case MLXPLAT_CPLD_LPC_REG_AGGRCO_OFFSET:
1702 case MLXPLAT_CPLD_LPC_REG_AGGRCO_MASK_OFFSET:
1703 case MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET:
1704 case MLXPLAT_CPLD_LPC_REG_ASIC_EVENT_OFFSET:
1705 case MLXPLAT_CPLD_LPC_REG_ASIC_MASK_OFFSET:
1706 case MLXPLAT_CPLD_LPC_REG_PSU_OFFSET:
1707 case MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET:
1708 case MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET:
1709 case MLXPLAT_CPLD_LPC_REG_PWR_OFFSET:
1710 case MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET:
1711 case MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET:
1712 case MLXPLAT_CPLD_LPC_REG_FAN_OFFSET:
1713 case MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET:
1714 case MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET:
1715 case MLXPLAT_CPLD_LPC_REG_WD2_TMR_OFFSET:
1716 case MLXPLAT_CPLD_LPC_REG_WD2_TLEFT_OFFSET:
1717 case MLXPLAT_CPLD_LPC_REG_WD3_TMR_OFFSET:
1718 case MLXPLAT_CPLD_LPC_REG_WD3_TLEFT_OFFSET:
1719 case MLXPLAT_CPLD_LPC_REG_PWM1_OFFSET:
1720 case MLXPLAT_CPLD_LPC_REG_TACHO1_OFFSET:
1721 case MLXPLAT_CPLD_LPC_REG_TACHO2_OFFSET:
1722 case MLXPLAT_CPLD_LPC_REG_TACHO3_OFFSET:
1723 case MLXPLAT_CPLD_LPC_REG_TACHO4_OFFSET:
1724 case MLXPLAT_CPLD_LPC_REG_TACHO5_OFFSET:
1725 case MLXPLAT_CPLD_LPC_REG_TACHO6_OFFSET:
1726 case MLXPLAT_CPLD_LPC_REG_TACHO7_OFFSET:
1727 case MLXPLAT_CPLD_LPC_REG_TACHO8_OFFSET:
1728 case MLXPLAT_CPLD_LPC_REG_TACHO9_OFFSET:
1729 case MLXPLAT_CPLD_LPC_REG_TACHO10_OFFSET:
1730 case MLXPLAT_CPLD_LPC_REG_TACHO11_OFFSET:
1731 case MLXPLAT_CPLD_LPC_REG_TACHO12_OFFSET:
1732 case MLXPLAT_CPLD_LPC_REG_PWM_CONTROL_OFFSET:
1733 case MLXPLAT_CPLD_LPC_REG_FAN_CAP1_OFFSET:
1734 case MLXPLAT_CPLD_LPC_REG_FAN_CAP2_OFFSET:
1735 case MLXPLAT_CPLD_LPC_REG_FAN_DRW_CAP_OFFSET:
1736 case MLXPLAT_CPLD_LPC_REG_TACHO_SPEED_OFFSET:
1737 return true;
1738 }
1739 return false;
1740 }
1741
1742 static const struct reg_default mlxplat_mlxcpld_regmap_default[] = {
1743 { MLXPLAT_CPLD_LPC_REG_WP1_OFFSET, 0x00 },
1744 { MLXPLAT_CPLD_LPC_REG_WP2_OFFSET, 0x00 },
1745 { MLXPLAT_CPLD_LPC_REG_PWM_CONTROL_OFFSET, 0x00 },
1746 { MLXPLAT_CPLD_LPC_REG_WD_CLEAR_WP_OFFSET, 0x00 },
1747 };
1748
1749 static const struct reg_default mlxplat_mlxcpld_regmap_ng[] = {
1750 { MLXPLAT_CPLD_LPC_REG_PWM_CONTROL_OFFSET, 0x00 },
1751 { MLXPLAT_CPLD_LPC_REG_WD_CLEAR_WP_OFFSET, 0x00 },
1752 };
1753
1754 struct mlxplat_mlxcpld_regmap_context {
1755 void __iomem *base;
1756 };
1757
1758 static struct mlxplat_mlxcpld_regmap_context mlxplat_mlxcpld_regmap_ctx;
1759
1760 static int
mlxplat_mlxcpld_reg_read(void * context,unsigned int reg,unsigned int * val)1761 mlxplat_mlxcpld_reg_read(void *context, unsigned int reg, unsigned int *val)
1762 {
1763 struct mlxplat_mlxcpld_regmap_context *ctx = context;
1764
1765 *val = ioread8(ctx->base + reg);
1766 return 0;
1767 }
1768
1769 static int
mlxplat_mlxcpld_reg_write(void * context,unsigned int reg,unsigned int val)1770 mlxplat_mlxcpld_reg_write(void *context, unsigned int reg, unsigned int val)
1771 {
1772 struct mlxplat_mlxcpld_regmap_context *ctx = context;
1773
1774 iowrite8(val, ctx->base + reg);
1775 return 0;
1776 }
1777
1778 static const struct regmap_config mlxplat_mlxcpld_regmap_config = {
1779 .reg_bits = 8,
1780 .val_bits = 8,
1781 .max_register = 255,
1782 .cache_type = REGCACHE_FLAT,
1783 .writeable_reg = mlxplat_mlxcpld_writeable_reg,
1784 .readable_reg = mlxplat_mlxcpld_readable_reg,
1785 .volatile_reg = mlxplat_mlxcpld_volatile_reg,
1786 .reg_defaults = mlxplat_mlxcpld_regmap_default,
1787 .num_reg_defaults = ARRAY_SIZE(mlxplat_mlxcpld_regmap_default),
1788 .reg_read = mlxplat_mlxcpld_reg_read,
1789 .reg_write = mlxplat_mlxcpld_reg_write,
1790 };
1791
1792 static const struct regmap_config mlxplat_mlxcpld_regmap_config_ng = {
1793 .reg_bits = 8,
1794 .val_bits = 8,
1795 .max_register = 255,
1796 .cache_type = REGCACHE_FLAT,
1797 .writeable_reg = mlxplat_mlxcpld_writeable_reg,
1798 .readable_reg = mlxplat_mlxcpld_readable_reg,
1799 .volatile_reg = mlxplat_mlxcpld_volatile_reg,
1800 .reg_defaults = mlxplat_mlxcpld_regmap_ng,
1801 .num_reg_defaults = ARRAY_SIZE(mlxplat_mlxcpld_regmap_ng),
1802 .reg_read = mlxplat_mlxcpld_reg_read,
1803 .reg_write = mlxplat_mlxcpld_reg_write,
1804 };
1805
1806 static struct resource mlxplat_mlxcpld_resources[] = {
1807 [0] = DEFINE_RES_IRQ_NAMED(17, "mlxreg-hotplug"),
1808 };
1809
1810 static struct platform_device *mlxplat_dev;
1811 static struct mlxreg_core_hotplug_platform_data *mlxplat_i2c;
1812 static struct mlxreg_core_hotplug_platform_data *mlxplat_hotplug;
1813 static struct mlxreg_core_platform_data *mlxplat_led;
1814 static struct mlxreg_core_platform_data *mlxplat_regs_io;
1815 static struct mlxreg_core_platform_data *mlxplat_fan;
1816 static struct mlxreg_core_platform_data
1817 *mlxplat_wd_data[MLXPLAT_CPLD_WD_MAX_DEVS];
1818 static const struct regmap_config *mlxplat_regmap_config;
1819
mlxplat_dmi_default_matched(const struct dmi_system_id * dmi)1820 static int __init mlxplat_dmi_default_matched(const struct dmi_system_id *dmi)
1821 {
1822 int i;
1823
1824 for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
1825 mlxplat_mux_data[i].values = mlxplat_default_channels[i];
1826 mlxplat_mux_data[i].n_values =
1827 ARRAY_SIZE(mlxplat_default_channels[i]);
1828 }
1829 mlxplat_hotplug = &mlxplat_mlxcpld_default_data;
1830 mlxplat_hotplug->deferred_nr =
1831 mlxplat_default_channels[i - 1][MLXPLAT_CPLD_GRP_CHNL_NUM - 1];
1832 mlxplat_led = &mlxplat_default_led_data;
1833 mlxplat_regs_io = &mlxplat_default_regs_io_data;
1834 mlxplat_wd_data[0] = &mlxplat_mlxcpld_wd_set_type1[0];
1835
1836 return 1;
1837 };
1838
mlxplat_dmi_msn21xx_matched(const struct dmi_system_id * dmi)1839 static int __init mlxplat_dmi_msn21xx_matched(const struct dmi_system_id *dmi)
1840 {
1841 int i;
1842
1843 for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
1844 mlxplat_mux_data[i].values = mlxplat_msn21xx_channels;
1845 mlxplat_mux_data[i].n_values =
1846 ARRAY_SIZE(mlxplat_msn21xx_channels);
1847 }
1848 mlxplat_hotplug = &mlxplat_mlxcpld_msn21xx_data;
1849 mlxplat_hotplug->deferred_nr =
1850 mlxplat_msn21xx_channels[MLXPLAT_CPLD_GRP_CHNL_NUM - 1];
1851 mlxplat_led = &mlxplat_msn21xx_led_data;
1852 mlxplat_regs_io = &mlxplat_msn21xx_regs_io_data;
1853 mlxplat_wd_data[0] = &mlxplat_mlxcpld_wd_set_type1[0];
1854
1855 return 1;
1856 };
1857
mlxplat_dmi_msn274x_matched(const struct dmi_system_id * dmi)1858 static int __init mlxplat_dmi_msn274x_matched(const struct dmi_system_id *dmi)
1859 {
1860 int i;
1861
1862 for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
1863 mlxplat_mux_data[i].values = mlxplat_msn21xx_channels;
1864 mlxplat_mux_data[i].n_values =
1865 ARRAY_SIZE(mlxplat_msn21xx_channels);
1866 }
1867 mlxplat_hotplug = &mlxplat_mlxcpld_msn274x_data;
1868 mlxplat_hotplug->deferred_nr =
1869 mlxplat_msn21xx_channels[MLXPLAT_CPLD_GRP_CHNL_NUM - 1];
1870 mlxplat_led = &mlxplat_default_led_data;
1871 mlxplat_regs_io = &mlxplat_msn21xx_regs_io_data;
1872 mlxplat_wd_data[0] = &mlxplat_mlxcpld_wd_set_type1[0];
1873
1874 return 1;
1875 };
1876
mlxplat_dmi_msn201x_matched(const struct dmi_system_id * dmi)1877 static int __init mlxplat_dmi_msn201x_matched(const struct dmi_system_id *dmi)
1878 {
1879 int i;
1880
1881 for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
1882 mlxplat_mux_data[i].values = mlxplat_msn21xx_channels;
1883 mlxplat_mux_data[i].n_values =
1884 ARRAY_SIZE(mlxplat_msn21xx_channels);
1885 }
1886 mlxplat_hotplug = &mlxplat_mlxcpld_msn201x_data;
1887 mlxplat_hotplug->deferred_nr =
1888 mlxplat_default_channels[i - 1][MLXPLAT_CPLD_GRP_CHNL_NUM - 1];
1889 mlxplat_led = &mlxplat_msn21xx_led_data;
1890 mlxplat_regs_io = &mlxplat_msn21xx_regs_io_data;
1891 mlxplat_wd_data[0] = &mlxplat_mlxcpld_wd_set_type1[0];
1892
1893 return 1;
1894 };
1895
mlxplat_dmi_qmb7xx_matched(const struct dmi_system_id * dmi)1896 static int __init mlxplat_dmi_qmb7xx_matched(const struct dmi_system_id *dmi)
1897 {
1898 int i;
1899
1900 for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
1901 mlxplat_mux_data[i].values = mlxplat_msn21xx_channels;
1902 mlxplat_mux_data[i].n_values =
1903 ARRAY_SIZE(mlxplat_msn21xx_channels);
1904 }
1905 mlxplat_hotplug = &mlxplat_mlxcpld_default_ng_data;
1906 mlxplat_hotplug->deferred_nr =
1907 mlxplat_msn21xx_channels[MLXPLAT_CPLD_GRP_CHNL_NUM - 1];
1908 mlxplat_led = &mlxplat_default_ng_led_data;
1909 mlxplat_regs_io = &mlxplat_default_ng_regs_io_data;
1910 mlxplat_fan = &mlxplat_default_fan_data;
1911 for (i = 0; i < ARRAY_SIZE(mlxplat_mlxcpld_wd_set_type2); i++)
1912 mlxplat_wd_data[i] = &mlxplat_mlxcpld_wd_set_type2[i];
1913 mlxplat_i2c = &mlxplat_mlxcpld_i2c_ng_data;
1914 mlxplat_regmap_config = &mlxplat_mlxcpld_regmap_config_ng;
1915
1916 return 1;
1917 };
1918
1919 static const struct dmi_system_id mlxplat_dmi_table[] __initconst = {
1920 {
1921 .callback = mlxplat_dmi_default_matched,
1922 .matches = {
1923 DMI_MATCH(DMI_BOARD_NAME, "VMOD0001"),
1924 },
1925 },
1926 {
1927 .callback = mlxplat_dmi_msn21xx_matched,
1928 .matches = {
1929 DMI_MATCH(DMI_BOARD_NAME, "VMOD0002"),
1930 },
1931 },
1932 {
1933 .callback = mlxplat_dmi_msn274x_matched,
1934 .matches = {
1935 DMI_MATCH(DMI_BOARD_NAME, "VMOD0003"),
1936 },
1937 },
1938 {
1939 .callback = mlxplat_dmi_msn201x_matched,
1940 .matches = {
1941 DMI_MATCH(DMI_BOARD_NAME, "VMOD0004"),
1942 },
1943 },
1944 {
1945 .callback = mlxplat_dmi_qmb7xx_matched,
1946 .matches = {
1947 DMI_MATCH(DMI_BOARD_NAME, "VMOD0005"),
1948 },
1949 },
1950 {
1951 .callback = mlxplat_dmi_qmb7xx_matched,
1952 .matches = {
1953 DMI_MATCH(DMI_BOARD_NAME, "VMOD0007"),
1954 },
1955 },
1956 {
1957 .callback = mlxplat_dmi_msn274x_matched,
1958 .matches = {
1959 DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
1960 DMI_MATCH(DMI_PRODUCT_NAME, "MSN274"),
1961 },
1962 },
1963 {
1964 .callback = mlxplat_dmi_default_matched,
1965 .matches = {
1966 DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
1967 DMI_MATCH(DMI_PRODUCT_NAME, "MSN24"),
1968 },
1969 },
1970 {
1971 .callback = mlxplat_dmi_default_matched,
1972 .matches = {
1973 DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
1974 DMI_MATCH(DMI_PRODUCT_NAME, "MSN27"),
1975 },
1976 },
1977 {
1978 .callback = mlxplat_dmi_default_matched,
1979 .matches = {
1980 DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
1981 DMI_MATCH(DMI_PRODUCT_NAME, "MSB"),
1982 },
1983 },
1984 {
1985 .callback = mlxplat_dmi_default_matched,
1986 .matches = {
1987 DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
1988 DMI_MATCH(DMI_PRODUCT_NAME, "MSX"),
1989 },
1990 },
1991 {
1992 .callback = mlxplat_dmi_msn21xx_matched,
1993 .matches = {
1994 DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
1995 DMI_MATCH(DMI_PRODUCT_NAME, "MSN21"),
1996 },
1997 },
1998 {
1999 .callback = mlxplat_dmi_msn201x_matched,
2000 .matches = {
2001 DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
2002 DMI_MATCH(DMI_PRODUCT_NAME, "MSN201"),
2003 },
2004 },
2005 {
2006 .callback = mlxplat_dmi_qmb7xx_matched,
2007 .matches = {
2008 DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
2009 DMI_MATCH(DMI_PRODUCT_NAME, "MQM87"),
2010 },
2011 },
2012 {
2013 .callback = mlxplat_dmi_qmb7xx_matched,
2014 .matches = {
2015 DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
2016 DMI_MATCH(DMI_PRODUCT_NAME, "MSN37"),
2017 },
2018 },
2019 {
2020 .callback = mlxplat_dmi_qmb7xx_matched,
2021 .matches = {
2022 DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
2023 DMI_MATCH(DMI_PRODUCT_NAME, "MSN34"),
2024 },
2025 },
2026 {
2027 .callback = mlxplat_dmi_qmb7xx_matched,
2028 .matches = {
2029 DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
2030 DMI_MATCH(DMI_PRODUCT_NAME, "MSN38"),
2031 },
2032 },
2033 { }
2034 };
2035
2036 MODULE_DEVICE_TABLE(dmi, mlxplat_dmi_table);
2037
mlxplat_mlxcpld_verify_bus_topology(int * nr)2038 static int mlxplat_mlxcpld_verify_bus_topology(int *nr)
2039 {
2040 struct i2c_adapter *search_adap;
2041 int shift, i;
2042
2043 /* Scan adapters from expected id to verify it is free. */
2044 *nr = MLXPLAT_CPLD_PHYS_ADAPTER_DEF_NR;
2045 for (i = MLXPLAT_CPLD_PHYS_ADAPTER_DEF_NR; i <
2046 MLXPLAT_CPLD_MAX_PHYS_ADAPTER_NUM; i++) {
2047 search_adap = i2c_get_adapter(i);
2048 if (search_adap) {
2049 i2c_put_adapter(search_adap);
2050 continue;
2051 }
2052
2053 /* Return if expected parent adapter is free. */
2054 if (i == MLXPLAT_CPLD_PHYS_ADAPTER_DEF_NR)
2055 return 0;
2056 break;
2057 }
2058
2059 /* Return with error if free id for adapter is not found. */
2060 if (i == MLXPLAT_CPLD_MAX_PHYS_ADAPTER_NUM)
2061 return -ENODEV;
2062
2063 /* Shift adapter ids, since expected parent adapter is not free. */
2064 *nr = i;
2065 for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
2066 shift = *nr - mlxplat_mux_data[i].parent;
2067 mlxplat_mux_data[i].parent = *nr;
2068 mlxplat_mux_data[i].base_nr += shift;
2069 if (shift > 0)
2070 mlxplat_hotplug->shift_nr = shift;
2071 }
2072
2073 return 0;
2074 }
2075
mlxplat_init(void)2076 static int __init mlxplat_init(void)
2077 {
2078 struct mlxplat_priv *priv;
2079 int i, j, nr, err;
2080
2081 if (!dmi_check_system(mlxplat_dmi_table))
2082 return -ENODEV;
2083
2084 mlxplat_dev = platform_device_register_simple(MLX_PLAT_DEVICE_NAME, -1,
2085 mlxplat_lpc_resources,
2086 ARRAY_SIZE(mlxplat_lpc_resources));
2087
2088 if (IS_ERR(mlxplat_dev))
2089 return PTR_ERR(mlxplat_dev);
2090
2091 priv = devm_kzalloc(&mlxplat_dev->dev, sizeof(struct mlxplat_priv),
2092 GFP_KERNEL);
2093 if (!priv) {
2094 err = -ENOMEM;
2095 goto fail_alloc;
2096 }
2097 platform_set_drvdata(mlxplat_dev, priv);
2098
2099 mlxplat_mlxcpld_regmap_ctx.base = devm_ioport_map(&mlxplat_dev->dev,
2100 mlxplat_lpc_resources[1].start, 1);
2101 if (!mlxplat_mlxcpld_regmap_ctx.base) {
2102 err = -ENOMEM;
2103 goto fail_alloc;
2104 }
2105
2106 if (!mlxplat_regmap_config)
2107 mlxplat_regmap_config = &mlxplat_mlxcpld_regmap_config;
2108
2109 priv->regmap = devm_regmap_init(&mlxplat_dev->dev, NULL,
2110 &mlxplat_mlxcpld_regmap_ctx,
2111 mlxplat_regmap_config);
2112 if (IS_ERR(priv->regmap)) {
2113 err = PTR_ERR(priv->regmap);
2114 goto fail_alloc;
2115 }
2116
2117 err = mlxplat_mlxcpld_verify_bus_topology(&nr);
2118 if (nr < 0)
2119 goto fail_alloc;
2120
2121 nr = (nr == MLXPLAT_CPLD_MAX_PHYS_ADAPTER_NUM) ? -1 : nr;
2122 if (mlxplat_i2c)
2123 mlxplat_i2c->regmap = priv->regmap;
2124 priv->pdev_i2c = platform_device_register_resndata(
2125 &mlxplat_dev->dev, "i2c_mlxcpld",
2126 nr, mlxplat_mlxcpld_resources,
2127 ARRAY_SIZE(mlxplat_mlxcpld_resources),
2128 mlxplat_i2c, sizeof(*mlxplat_i2c));
2129 if (IS_ERR(priv->pdev_i2c)) {
2130 err = PTR_ERR(priv->pdev_i2c);
2131 goto fail_alloc;
2132 }
2133
2134 for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
2135 priv->pdev_mux[i] = platform_device_register_resndata(
2136 &priv->pdev_i2c->dev,
2137 "i2c-mux-reg", i, NULL,
2138 0, &mlxplat_mux_data[i],
2139 sizeof(mlxplat_mux_data[i]));
2140 if (IS_ERR(priv->pdev_mux[i])) {
2141 err = PTR_ERR(priv->pdev_mux[i]);
2142 goto fail_platform_mux_register;
2143 }
2144 }
2145
2146 /* Add hotplug driver */
2147 mlxplat_hotplug->regmap = priv->regmap;
2148 priv->pdev_hotplug = platform_device_register_resndata(
2149 &mlxplat_dev->dev, "mlxreg-hotplug",
2150 PLATFORM_DEVID_NONE,
2151 mlxplat_mlxcpld_resources,
2152 ARRAY_SIZE(mlxplat_mlxcpld_resources),
2153 mlxplat_hotplug, sizeof(*mlxplat_hotplug));
2154 if (IS_ERR(priv->pdev_hotplug)) {
2155 err = PTR_ERR(priv->pdev_hotplug);
2156 goto fail_platform_mux_register;
2157 }
2158
2159 /* Set default registers. */
2160 for (j = 0; j < mlxplat_regmap_config->num_reg_defaults; j++) {
2161 err = regmap_write(priv->regmap,
2162 mlxplat_regmap_config->reg_defaults[j].reg,
2163 mlxplat_regmap_config->reg_defaults[j].def);
2164 if (err)
2165 goto fail_platform_mux_register;
2166 }
2167
2168 /* Add LED driver. */
2169 mlxplat_led->regmap = priv->regmap;
2170 priv->pdev_led = platform_device_register_resndata(
2171 &mlxplat_dev->dev, "leds-mlxreg",
2172 PLATFORM_DEVID_NONE, NULL, 0,
2173 mlxplat_led, sizeof(*mlxplat_led));
2174 if (IS_ERR(priv->pdev_led)) {
2175 err = PTR_ERR(priv->pdev_led);
2176 goto fail_platform_hotplug_register;
2177 }
2178
2179 /* Add registers io access driver. */
2180 if (mlxplat_regs_io) {
2181 mlxplat_regs_io->regmap = priv->regmap;
2182 priv->pdev_io_regs = platform_device_register_resndata(
2183 &mlxplat_dev->dev, "mlxreg-io",
2184 PLATFORM_DEVID_NONE, NULL, 0,
2185 mlxplat_regs_io,
2186 sizeof(*mlxplat_regs_io));
2187 if (IS_ERR(priv->pdev_io_regs)) {
2188 err = PTR_ERR(priv->pdev_io_regs);
2189 goto fail_platform_led_register;
2190 }
2191 }
2192
2193 /* Add FAN driver. */
2194 if (mlxplat_fan) {
2195 mlxplat_fan->regmap = priv->regmap;
2196 priv->pdev_fan = platform_device_register_resndata(
2197 &mlxplat_dev->dev, "mlxreg-fan",
2198 PLATFORM_DEVID_NONE, NULL, 0,
2199 mlxplat_fan,
2200 sizeof(*mlxplat_fan));
2201 if (IS_ERR(priv->pdev_fan)) {
2202 err = PTR_ERR(priv->pdev_fan);
2203 goto fail_platform_io_regs_register;
2204 }
2205 }
2206
2207 /* Add WD drivers. */
2208 for (j = 0; j < MLXPLAT_CPLD_WD_MAX_DEVS; j++) {
2209 if (mlxplat_wd_data[j]) {
2210 mlxplat_wd_data[j]->regmap = priv->regmap;
2211 priv->pdev_wd[j] = platform_device_register_resndata(
2212 &mlxplat_dev->dev, "mlx-wdt",
2213 j, NULL, 0,
2214 mlxplat_wd_data[j],
2215 sizeof(*mlxplat_wd_data[j]));
2216 if (IS_ERR(priv->pdev_wd[j])) {
2217 err = PTR_ERR(priv->pdev_wd[j]);
2218 goto fail_platform_wd_register;
2219 }
2220 }
2221 }
2222
2223 /* Sync registers with hardware. */
2224 regcache_mark_dirty(priv->regmap);
2225 err = regcache_sync(priv->regmap);
2226 if (err)
2227 goto fail_platform_wd_register;
2228
2229 return 0;
2230
2231 fail_platform_wd_register:
2232 while (--j >= 0)
2233 platform_device_unregister(priv->pdev_wd[j]);
2234 if (mlxplat_fan)
2235 platform_device_unregister(priv->pdev_fan);
2236 fail_platform_io_regs_register:
2237 if (mlxplat_regs_io)
2238 platform_device_unregister(priv->pdev_io_regs);
2239 fail_platform_led_register:
2240 platform_device_unregister(priv->pdev_led);
2241 fail_platform_hotplug_register:
2242 platform_device_unregister(priv->pdev_hotplug);
2243 fail_platform_mux_register:
2244 while (--i >= 0)
2245 platform_device_unregister(priv->pdev_mux[i]);
2246 platform_device_unregister(priv->pdev_i2c);
2247 fail_alloc:
2248 platform_device_unregister(mlxplat_dev);
2249
2250 return err;
2251 }
2252 module_init(mlxplat_init);
2253
mlxplat_exit(void)2254 static void __exit mlxplat_exit(void)
2255 {
2256 struct mlxplat_priv *priv = platform_get_drvdata(mlxplat_dev);
2257 int i;
2258
2259 for (i = MLXPLAT_CPLD_WD_MAX_DEVS - 1; i >= 0 ; i--)
2260 platform_device_unregister(priv->pdev_wd[i]);
2261 if (priv->pdev_fan)
2262 platform_device_unregister(priv->pdev_fan);
2263 if (priv->pdev_io_regs)
2264 platform_device_unregister(priv->pdev_io_regs);
2265 platform_device_unregister(priv->pdev_led);
2266 platform_device_unregister(priv->pdev_hotplug);
2267
2268 for (i = ARRAY_SIZE(mlxplat_mux_data) - 1; i >= 0 ; i--)
2269 platform_device_unregister(priv->pdev_mux[i]);
2270
2271 platform_device_unregister(priv->pdev_i2c);
2272 platform_device_unregister(mlxplat_dev);
2273 }
2274 module_exit(mlxplat_exit);
2275
2276 MODULE_AUTHOR("Vadim Pasternak (vadimp@mellanox.com)");
2277 MODULE_DESCRIPTION("Mellanox platform driver");
2278 MODULE_LICENSE("Dual BSD/GPL");
2279