mlx-platform.c 48 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740
  1. /*
  2. * Copyright (c) 2016 Mellanox Technologies. All rights reserved.
  3. * Copyright (c) 2016 Vadim Pasternak <vadimp@mellanox.com>
  4. *
  5. * Redistribution and use in source and binary forms, with or without
  6. * modification, are permitted provided that the following conditions are met:
  7. *
  8. * 1. Redistributions of source code must retain the above copyright
  9. * notice, this list of conditions and the following disclaimer.
  10. * 2. Redistributions in binary form must reproduce the above copyright
  11. * notice, this list of conditions and the following disclaimer in the
  12. * documentation and/or other materials provided with the distribution.
  13. * 3. Neither the names of the copyright holders nor the names of its
  14. * contributors may be used to endorse or promote products derived from
  15. * this software without specific prior written permission.
  16. *
  17. * Alternatively, this software may be distributed under the terms of the
  18. * GNU General Public License ("GPL") version 2 as published by the Free
  19. * Software Foundation.
  20. *
  21. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  22. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  23. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  24. * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
  25. * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
  26. * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
  27. * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
  28. * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
  29. * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  30. * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  31. * POSSIBILITY OF SUCH DAMAGE.
  32. */
  33. #include <linux/device.h>
  34. #include <linux/dmi.h>
  35. #include <linux/i2c.h>
  36. #include <linux/i2c-mux.h>
  37. #include <linux/io.h>
  38. #include <linux/module.h>
  39. #include <linux/platform_device.h>
  40. #include <linux/platform_data/i2c-mux-reg.h>
  41. #include <linux/platform_data/mlxreg.h>
  42. #include <linux/regmap.h>
  43. #define MLX_PLAT_DEVICE_NAME "mlxplat"
  44. /* LPC bus IO offsets */
  45. #define MLXPLAT_CPLD_LPC_I2C_BASE_ADRR 0x2000
  46. #define MLXPLAT_CPLD_LPC_REG_BASE_ADRR 0x2500
  47. #define MLXPLAT_CPLD_LPC_REG_CPLD1_VER_OFFSET 0x00
  48. #define MLXPLAT_CPLD_LPC_REG_CPLD2_VER_OFFSET 0x01
  49. #define MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET 0x1d
  50. #define MLXPLAT_CPLD_LPC_REG_LED1_OFFSET 0x20
  51. #define MLXPLAT_CPLD_LPC_REG_LED2_OFFSET 0x21
  52. #define MLXPLAT_CPLD_LPC_REG_LED3_OFFSET 0x22
  53. #define MLXPLAT_CPLD_LPC_REG_LED4_OFFSET 0x23
  54. #define MLXPLAT_CPLD_LPC_REG_LED5_OFFSET 0x24
  55. #define MLXPLAT_CPLD_LPC_REG_GP1_OFFSET 0x30
  56. #define MLXPLAT_CPLD_LPC_REG_WP1_OFFSET 0x31
  57. #define MLXPLAT_CPLD_LPC_REG_GP2_OFFSET 0x32
  58. #define MLXPLAT_CPLD_LPC_REG_WP2_OFFSET 0x33
  59. #define MLXPLAT_CPLD_LPC_REG_PWM_CONTROL_OFFSET 0x37
  60. #define MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET 0x3a
  61. #define MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET 0x3b
  62. #define MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET 0x40
  63. #define MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET 0x41
  64. #define MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET 0x50
  65. #define MLXPLAT_CPLD_LPC_REG_ASIC_EVENT_OFFSET 0x51
  66. #define MLXPLAT_CPLD_LPC_REG_ASIC_MASK_OFFSET 0x52
  67. #define MLXPLAT_CPLD_LPC_REG_PSU_OFFSET 0x58
  68. #define MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET 0x59
  69. #define MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET 0x5a
  70. #define MLXPLAT_CPLD_LPC_REG_PWR_OFFSET 0x64
  71. #define MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET 0x65
  72. #define MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET 0x66
  73. #define MLXPLAT_CPLD_LPC_REG_FAN_OFFSET 0x88
  74. #define MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET 0x89
  75. #define MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET 0x8a
  76. #define MLXPLAT_CPLD_LPC_REG_PWM1_OFFSET 0xe3
  77. #define MLXPLAT_CPLD_LPC_REG_TACHO1_OFFSET 0xe4
  78. #define MLXPLAT_CPLD_LPC_REG_TACHO2_OFFSET 0xe5
  79. #define MLXPLAT_CPLD_LPC_REG_TACHO3_OFFSET 0xe6
  80. #define MLXPLAT_CPLD_LPC_REG_TACHO4_OFFSET 0xe7
  81. #define MLXPLAT_CPLD_LPC_REG_TACHO5_OFFSET 0xe8
  82. #define MLXPLAT_CPLD_LPC_REG_TACHO6_OFFSET 0xe9
  83. #define MLXPLAT_CPLD_LPC_REG_TACHO7_OFFSET 0xeb
  84. #define MLXPLAT_CPLD_LPC_REG_TACHO8_OFFSET 0xec
  85. #define MLXPLAT_CPLD_LPC_REG_TACHO9_OFFSET 0xed
  86. #define MLXPLAT_CPLD_LPC_REG_TACHO10_OFFSET 0xee
  87. #define MLXPLAT_CPLD_LPC_REG_TACHO11_OFFSET 0xef
  88. #define MLXPLAT_CPLD_LPC_REG_TACHO12_OFFSET 0xf0
  89. #define MLXPLAT_CPLD_LPC_IO_RANGE 0x100
  90. #define MLXPLAT_CPLD_LPC_I2C_CH1_OFF 0xdb
  91. #define MLXPLAT_CPLD_LPC_I2C_CH2_OFF 0xda
  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. /* Masks for aggregation, psu, pwr and fan event in CPLD related registers. */
  100. #define MLXPLAT_CPLD_AGGR_ASIC_MASK_DEF 0x04
  101. #define MLXPLAT_CPLD_AGGR_PSU_MASK_DEF 0x08
  102. #define MLXPLAT_CPLD_AGGR_PWR_MASK_DEF 0x08
  103. #define MLXPLAT_CPLD_AGGR_FAN_MASK_DEF 0x40
  104. #define MLXPLAT_CPLD_AGGR_MASK_DEF (MLXPLAT_CPLD_AGGR_ASIC_MASK_DEF | \
  105. MLXPLAT_CPLD_AGGR_PSU_MASK_DEF | \
  106. MLXPLAT_CPLD_AGGR_FAN_MASK_DEF)
  107. #define MLXPLAT_CPLD_AGGR_ASIC_MASK_NG 0x01
  108. #define MLXPLAT_CPLD_AGGR_MASK_NG_DEF 0x04
  109. #define MLXPLAT_CPLD_LOW_AGGR_MASK_LOW 0xc1
  110. #define MLXPLAT_CPLD_PSU_MASK GENMASK(1, 0)
  111. #define MLXPLAT_CPLD_PWR_MASK GENMASK(1, 0)
  112. #define MLXPLAT_CPLD_FAN_MASK GENMASK(3, 0)
  113. #define MLXPLAT_CPLD_ASIC_MASK GENMASK(1, 0)
  114. #define MLXPLAT_CPLD_FAN_NG_MASK GENMASK(5, 0)
  115. #define MLXPLAT_CPLD_LED_LO_NIBBLE_MASK GENMASK(7, 4)
  116. #define MLXPLAT_CPLD_LED_HI_NIBBLE_MASK GENMASK(3, 0)
  117. /* Default I2C parent bus number */
  118. #define MLXPLAT_CPLD_PHYS_ADAPTER_DEF_NR 1
  119. /* Maximum number of possible physical buses equipped on system */
  120. #define MLXPLAT_CPLD_MAX_PHYS_ADAPTER_NUM 16
  121. /* Number of channels in group */
  122. #define MLXPLAT_CPLD_GRP_CHNL_NUM 8
  123. /* Start channel numbers */
  124. #define MLXPLAT_CPLD_CH1 2
  125. #define MLXPLAT_CPLD_CH2 10
  126. /* Number of LPC attached MUX platform devices */
  127. #define MLXPLAT_CPLD_LPC_MUX_DEVS 2
  128. /* Hotplug devices adapter numbers */
  129. #define MLXPLAT_CPLD_NR_NONE -1
  130. #define MLXPLAT_CPLD_PSU_DEFAULT_NR 10
  131. #define MLXPLAT_CPLD_PSU_MSNXXXX_NR 4
  132. #define MLXPLAT_CPLD_FAN1_DEFAULT_NR 11
  133. #define MLXPLAT_CPLD_FAN2_DEFAULT_NR 12
  134. #define MLXPLAT_CPLD_FAN3_DEFAULT_NR 13
  135. #define MLXPLAT_CPLD_FAN4_DEFAULT_NR 14
  136. /* mlxplat_priv - platform private data
  137. * @pdev_i2c - i2c controller platform device
  138. * @pdev_mux - array of mux platform devices
  139. * @pdev_hotplug - hotplug platform devices
  140. * @pdev_led - led platform devices
  141. * @pdev_io_regs - register access platform devices
  142. * @pdev_fan - FAN platform devices
  143. */
  144. struct mlxplat_priv {
  145. struct platform_device *pdev_i2c;
  146. struct platform_device *pdev_mux[MLXPLAT_CPLD_LPC_MUX_DEVS];
  147. struct platform_device *pdev_hotplug;
  148. struct platform_device *pdev_led;
  149. struct platform_device *pdev_io_regs;
  150. struct platform_device *pdev_fan;
  151. };
  152. /* Regions for LPC I2C controller and LPC base register space */
  153. static const struct resource mlxplat_lpc_resources[] = {
  154. [0] = DEFINE_RES_NAMED(MLXPLAT_CPLD_LPC_I2C_BASE_ADRR,
  155. MLXPLAT_CPLD_LPC_IO_RANGE,
  156. "mlxplat_cpld_lpc_i2c_ctrl", IORESOURCE_IO),
  157. [1] = DEFINE_RES_NAMED(MLXPLAT_CPLD_LPC_REG_BASE_ADRR,
  158. MLXPLAT_CPLD_LPC_IO_RANGE,
  159. "mlxplat_cpld_lpc_regs",
  160. IORESOURCE_IO),
  161. };
  162. /* Platform default channels */
  163. static const int mlxplat_default_channels[][MLXPLAT_CPLD_GRP_CHNL_NUM] = {
  164. {
  165. MLXPLAT_CPLD_CH1, MLXPLAT_CPLD_CH1 + 1, MLXPLAT_CPLD_CH1 + 2,
  166. MLXPLAT_CPLD_CH1 + 3, MLXPLAT_CPLD_CH1 + 4, MLXPLAT_CPLD_CH1 +
  167. 5, MLXPLAT_CPLD_CH1 + 6, MLXPLAT_CPLD_CH1 + 7
  168. },
  169. {
  170. MLXPLAT_CPLD_CH2, MLXPLAT_CPLD_CH2 + 1, MLXPLAT_CPLD_CH2 + 2,
  171. MLXPLAT_CPLD_CH2 + 3, MLXPLAT_CPLD_CH2 + 4, MLXPLAT_CPLD_CH2 +
  172. 5, MLXPLAT_CPLD_CH2 + 6, MLXPLAT_CPLD_CH2 + 7
  173. },
  174. };
  175. /* Platform channels for MSN21xx system family */
  176. static const int mlxplat_msn21xx_channels[] = { 1, 2, 3, 4, 5, 6, 7, 8 };
  177. /* Platform mux data */
  178. static struct i2c_mux_reg_platform_data mlxplat_mux_data[] = {
  179. {
  180. .parent = 1,
  181. .base_nr = MLXPLAT_CPLD_CH1,
  182. .write_only = 1,
  183. .reg = (void __iomem *)MLXPLAT_CPLD_LPC_REG1,
  184. .reg_size = 1,
  185. .idle_in_use = 1,
  186. },
  187. {
  188. .parent = 1,
  189. .base_nr = MLXPLAT_CPLD_CH2,
  190. .write_only = 1,
  191. .reg = (void __iomem *)MLXPLAT_CPLD_LPC_REG2,
  192. .reg_size = 1,
  193. .idle_in_use = 1,
  194. },
  195. };
  196. /* Platform hotplug devices */
  197. static struct i2c_board_info mlxplat_mlxcpld_pwr[] = {
  198. {
  199. I2C_BOARD_INFO("dps460", 0x59),
  200. },
  201. {
  202. I2C_BOARD_INFO("dps460", 0x58),
  203. },
  204. };
  205. static struct i2c_board_info mlxplat_mlxcpld_fan[] = {
  206. {
  207. I2C_BOARD_INFO("24c32", 0x50),
  208. },
  209. {
  210. I2C_BOARD_INFO("24c32", 0x50),
  211. },
  212. {
  213. I2C_BOARD_INFO("24c32", 0x50),
  214. },
  215. {
  216. I2C_BOARD_INFO("24c32", 0x50),
  217. },
  218. };
  219. /* Platform hotplug default data */
  220. static struct mlxreg_core_data mlxplat_mlxcpld_default_psu_items_data[] = {
  221. {
  222. .label = "psu1",
  223. .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
  224. .mask = BIT(0),
  225. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  226. },
  227. {
  228. .label = "psu2",
  229. .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
  230. .mask = BIT(1),
  231. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  232. },
  233. };
  234. static struct mlxreg_core_data mlxplat_mlxcpld_default_pwr_items_data[] = {
  235. {
  236. .label = "pwr1",
  237. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  238. .mask = BIT(0),
  239. .hpdev.brdinfo = &mlxplat_mlxcpld_pwr[0],
  240. .hpdev.nr = MLXPLAT_CPLD_PSU_DEFAULT_NR,
  241. },
  242. {
  243. .label = "pwr2",
  244. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  245. .mask = BIT(1),
  246. .hpdev.brdinfo = &mlxplat_mlxcpld_pwr[1],
  247. .hpdev.nr = MLXPLAT_CPLD_PSU_DEFAULT_NR,
  248. },
  249. };
  250. static struct mlxreg_core_data mlxplat_mlxcpld_default_fan_items_data[] = {
  251. {
  252. .label = "fan1",
  253. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  254. .mask = BIT(0),
  255. .hpdev.brdinfo = &mlxplat_mlxcpld_fan[0],
  256. .hpdev.nr = MLXPLAT_CPLD_FAN1_DEFAULT_NR,
  257. },
  258. {
  259. .label = "fan2",
  260. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  261. .mask = BIT(1),
  262. .hpdev.brdinfo = &mlxplat_mlxcpld_fan[1],
  263. .hpdev.nr = MLXPLAT_CPLD_FAN2_DEFAULT_NR,
  264. },
  265. {
  266. .label = "fan3",
  267. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  268. .mask = BIT(2),
  269. .hpdev.brdinfo = &mlxplat_mlxcpld_fan[2],
  270. .hpdev.nr = MLXPLAT_CPLD_FAN3_DEFAULT_NR,
  271. },
  272. {
  273. .label = "fan4",
  274. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  275. .mask = BIT(3),
  276. .hpdev.brdinfo = &mlxplat_mlxcpld_fan[3],
  277. .hpdev.nr = MLXPLAT_CPLD_FAN4_DEFAULT_NR,
  278. },
  279. };
  280. static struct mlxreg_core_data mlxplat_mlxcpld_default_asic_items_data[] = {
  281. {
  282. .label = "asic1",
  283. .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
  284. .mask = MLXPLAT_CPLD_ASIC_MASK,
  285. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  286. },
  287. };
  288. static struct mlxreg_core_item mlxplat_mlxcpld_default_items[] = {
  289. {
  290. .data = mlxplat_mlxcpld_default_psu_items_data,
  291. .aggr_mask = MLXPLAT_CPLD_AGGR_PSU_MASK_DEF,
  292. .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
  293. .mask = MLXPLAT_CPLD_PSU_MASK,
  294. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_psu_items_data),
  295. .inversed = 1,
  296. .health = false,
  297. },
  298. {
  299. .data = mlxplat_mlxcpld_default_pwr_items_data,
  300. .aggr_mask = MLXPLAT_CPLD_AGGR_PWR_MASK_DEF,
  301. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  302. .mask = MLXPLAT_CPLD_PWR_MASK,
  303. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_pwr_items_data),
  304. .inversed = 0,
  305. .health = false,
  306. },
  307. {
  308. .data = mlxplat_mlxcpld_default_fan_items_data,
  309. .aggr_mask = MLXPLAT_CPLD_AGGR_FAN_MASK_DEF,
  310. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  311. .mask = MLXPLAT_CPLD_FAN_MASK,
  312. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_fan_items_data),
  313. .inversed = 1,
  314. .health = false,
  315. },
  316. {
  317. .data = mlxplat_mlxcpld_default_asic_items_data,
  318. .aggr_mask = MLXPLAT_CPLD_AGGR_ASIC_MASK_DEF,
  319. .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
  320. .mask = MLXPLAT_CPLD_ASIC_MASK,
  321. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_asic_items_data),
  322. .inversed = 0,
  323. .health = true,
  324. },
  325. };
  326. static
  327. struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_default_data = {
  328. .items = mlxplat_mlxcpld_default_items,
  329. .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_items),
  330. .cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET,
  331. .mask = MLXPLAT_CPLD_AGGR_MASK_DEF,
  332. .cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET,
  333. .mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW,
  334. };
  335. static struct mlxreg_core_data mlxplat_mlxcpld_msn21xx_pwr_items_data[] = {
  336. {
  337. .label = "pwr1",
  338. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  339. .mask = BIT(0),
  340. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  341. },
  342. {
  343. .label = "pwr2",
  344. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  345. .mask = BIT(1),
  346. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  347. },
  348. };
  349. /* Platform hotplug MSN21xx system family data */
  350. static struct mlxreg_core_item mlxplat_mlxcpld_msn21xx_items[] = {
  351. {
  352. .data = mlxplat_mlxcpld_msn21xx_pwr_items_data,
  353. .aggr_mask = MLXPLAT_CPLD_AGGR_PWR_MASK_DEF,
  354. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  355. .mask = MLXPLAT_CPLD_PWR_MASK,
  356. .count = ARRAY_SIZE(mlxplat_mlxcpld_msn21xx_pwr_items_data),
  357. .inversed = 0,
  358. .health = false,
  359. },
  360. {
  361. .data = mlxplat_mlxcpld_default_asic_items_data,
  362. .aggr_mask = MLXPLAT_CPLD_AGGR_ASIC_MASK_DEF,
  363. .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
  364. .mask = MLXPLAT_CPLD_ASIC_MASK,
  365. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_asic_items_data),
  366. .inversed = 0,
  367. .health = true,
  368. },
  369. };
  370. static
  371. struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_msn21xx_data = {
  372. .items = mlxplat_mlxcpld_msn21xx_items,
  373. .counter = ARRAY_SIZE(mlxplat_mlxcpld_msn21xx_items),
  374. .cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET,
  375. .mask = MLXPLAT_CPLD_AGGR_MASK_DEF,
  376. .cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET,
  377. .mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW,
  378. };
  379. /* Platform hotplug msn274x system family data */
  380. static struct mlxreg_core_data mlxplat_mlxcpld_msn274x_psu_items_data[] = {
  381. {
  382. .label = "psu1",
  383. .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
  384. .mask = BIT(0),
  385. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  386. },
  387. {
  388. .label = "psu2",
  389. .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
  390. .mask = BIT(1),
  391. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  392. },
  393. };
  394. static struct mlxreg_core_data mlxplat_mlxcpld_default_ng_pwr_items_data[] = {
  395. {
  396. .label = "pwr1",
  397. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  398. .mask = BIT(0),
  399. .hpdev.brdinfo = &mlxplat_mlxcpld_pwr[0],
  400. .hpdev.nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR,
  401. },
  402. {
  403. .label = "pwr2",
  404. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  405. .mask = BIT(1),
  406. .hpdev.brdinfo = &mlxplat_mlxcpld_pwr[1],
  407. .hpdev.nr = MLXPLAT_CPLD_PSU_MSNXXXX_NR,
  408. },
  409. };
  410. static struct mlxreg_core_data mlxplat_mlxcpld_msn274x_fan_items_data[] = {
  411. {
  412. .label = "fan1",
  413. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  414. .mask = BIT(0),
  415. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  416. },
  417. {
  418. .label = "fan2",
  419. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  420. .mask = BIT(1),
  421. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  422. },
  423. {
  424. .label = "fan3",
  425. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  426. .mask = BIT(2),
  427. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  428. },
  429. {
  430. .label = "fan4",
  431. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  432. .mask = BIT(3),
  433. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  434. },
  435. };
  436. static struct mlxreg_core_item mlxplat_mlxcpld_msn274x_items[] = {
  437. {
  438. .data = mlxplat_mlxcpld_msn274x_psu_items_data,
  439. .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
  440. .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
  441. .mask = MLXPLAT_CPLD_PSU_MASK,
  442. .count = ARRAY_SIZE(mlxplat_mlxcpld_msn274x_psu_items_data),
  443. .inversed = 1,
  444. .health = false,
  445. },
  446. {
  447. .data = mlxplat_mlxcpld_default_ng_pwr_items_data,
  448. .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
  449. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  450. .mask = MLXPLAT_CPLD_PWR_MASK,
  451. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_pwr_items_data),
  452. .inversed = 0,
  453. .health = false,
  454. },
  455. {
  456. .data = mlxplat_mlxcpld_msn274x_fan_items_data,
  457. .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
  458. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  459. .mask = MLXPLAT_CPLD_FAN_MASK,
  460. .count = ARRAY_SIZE(mlxplat_mlxcpld_msn274x_fan_items_data),
  461. .inversed = 1,
  462. .health = false,
  463. },
  464. {
  465. .data = mlxplat_mlxcpld_default_asic_items_data,
  466. .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
  467. .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
  468. .mask = MLXPLAT_CPLD_ASIC_MASK,
  469. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_asic_items_data),
  470. .inversed = 0,
  471. .health = true,
  472. },
  473. };
  474. static
  475. struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_msn274x_data = {
  476. .items = mlxplat_mlxcpld_msn274x_items,
  477. .counter = ARRAY_SIZE(mlxplat_mlxcpld_msn274x_items),
  478. .cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET,
  479. .mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
  480. .cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET,
  481. .mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW,
  482. };
  483. /* Platform hotplug MSN201x system family data */
  484. static struct mlxreg_core_data mlxplat_mlxcpld_msn201x_pwr_items_data[] = {
  485. {
  486. .label = "pwr1",
  487. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  488. .mask = BIT(0),
  489. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  490. },
  491. {
  492. .label = "pwr2",
  493. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  494. .mask = BIT(1),
  495. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  496. },
  497. };
  498. static struct mlxreg_core_item mlxplat_mlxcpld_msn201x_items[] = {
  499. {
  500. .data = mlxplat_mlxcpld_msn201x_pwr_items_data,
  501. .aggr_mask = MLXPLAT_CPLD_AGGR_PWR_MASK_DEF,
  502. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  503. .mask = MLXPLAT_CPLD_PWR_MASK,
  504. .count = ARRAY_SIZE(mlxplat_mlxcpld_msn201x_pwr_items_data),
  505. .inversed = 0,
  506. .health = false,
  507. },
  508. {
  509. .data = mlxplat_mlxcpld_default_asic_items_data,
  510. .aggr_mask = MLXPLAT_CPLD_AGGR_ASIC_MASK_DEF,
  511. .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
  512. .mask = MLXPLAT_CPLD_ASIC_MASK,
  513. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_asic_items_data),
  514. .inversed = 0,
  515. .health = true,
  516. },
  517. };
  518. static
  519. struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_msn201x_data = {
  520. .items = mlxplat_mlxcpld_msn201x_items,
  521. .counter = ARRAY_SIZE(mlxplat_mlxcpld_msn201x_items),
  522. .cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET,
  523. .mask = MLXPLAT_CPLD_AGGR_MASK_DEF,
  524. .cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET,
  525. .mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW,
  526. };
  527. /* Platform hotplug next generation system family data */
  528. static struct mlxreg_core_data mlxplat_mlxcpld_default_ng_psu_items_data[] = {
  529. {
  530. .label = "psu1",
  531. .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
  532. .mask = BIT(0),
  533. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  534. },
  535. {
  536. .label = "psu2",
  537. .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
  538. .mask = BIT(1),
  539. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  540. },
  541. };
  542. static struct mlxreg_core_data mlxplat_mlxcpld_default_ng_fan_items_data[] = {
  543. {
  544. .label = "fan1",
  545. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  546. .mask = BIT(0),
  547. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  548. },
  549. {
  550. .label = "fan2",
  551. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  552. .mask = BIT(1),
  553. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  554. },
  555. {
  556. .label = "fan3",
  557. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  558. .mask = BIT(2),
  559. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  560. },
  561. {
  562. .label = "fan4",
  563. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  564. .mask = BIT(3),
  565. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  566. },
  567. {
  568. .label = "fan5",
  569. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  570. .mask = BIT(4),
  571. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  572. },
  573. {
  574. .label = "fan6",
  575. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  576. .mask = BIT(5),
  577. .hpdev.nr = MLXPLAT_CPLD_NR_NONE,
  578. },
  579. };
  580. static struct mlxreg_core_item mlxplat_mlxcpld_default_ng_items[] = {
  581. {
  582. .data = mlxplat_mlxcpld_default_ng_psu_items_data,
  583. .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
  584. .reg = MLXPLAT_CPLD_LPC_REG_PSU_OFFSET,
  585. .mask = MLXPLAT_CPLD_PSU_MASK,
  586. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_psu_items_data),
  587. .inversed = 1,
  588. .health = false,
  589. },
  590. {
  591. .data = mlxplat_mlxcpld_default_ng_pwr_items_data,
  592. .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
  593. .reg = MLXPLAT_CPLD_LPC_REG_PWR_OFFSET,
  594. .mask = MLXPLAT_CPLD_PWR_MASK,
  595. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_pwr_items_data),
  596. .inversed = 0,
  597. .health = false,
  598. },
  599. {
  600. .data = mlxplat_mlxcpld_default_ng_fan_items_data,
  601. .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
  602. .reg = MLXPLAT_CPLD_LPC_REG_FAN_OFFSET,
  603. .mask = MLXPLAT_CPLD_FAN_NG_MASK,
  604. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_fan_items_data),
  605. .inversed = 1,
  606. .health = false,
  607. },
  608. {
  609. .data = mlxplat_mlxcpld_default_asic_items_data,
  610. .aggr_mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
  611. .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
  612. .mask = MLXPLAT_CPLD_ASIC_MASK,
  613. .count = ARRAY_SIZE(mlxplat_mlxcpld_default_asic_items_data),
  614. .inversed = 0,
  615. .health = true,
  616. },
  617. };
  618. static
  619. struct mlxreg_core_hotplug_platform_data mlxplat_mlxcpld_default_ng_data = {
  620. .items = mlxplat_mlxcpld_default_ng_items,
  621. .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_items),
  622. .cell = MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET,
  623. .mask = MLXPLAT_CPLD_AGGR_MASK_NG_DEF,
  624. .cell_low = MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET,
  625. .mask_low = MLXPLAT_CPLD_LOW_AGGR_MASK_LOW,
  626. };
  627. /* Platform led default data */
  628. static struct mlxreg_core_data mlxplat_mlxcpld_default_led_data[] = {
  629. {
  630. .label = "status:green",
  631. .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
  632. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  633. },
  634. {
  635. .label = "status:red",
  636. .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
  637. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK
  638. },
  639. {
  640. .label = "psu:green",
  641. .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
  642. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  643. },
  644. {
  645. .label = "psu:red",
  646. .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
  647. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  648. },
  649. {
  650. .label = "fan1:green",
  651. .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
  652. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  653. },
  654. {
  655. .label = "fan1:red",
  656. .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
  657. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  658. },
  659. {
  660. .label = "fan2:green",
  661. .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
  662. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  663. },
  664. {
  665. .label = "fan2:red",
  666. .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
  667. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  668. },
  669. {
  670. .label = "fan3:green",
  671. .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
  672. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  673. },
  674. {
  675. .label = "fan3:red",
  676. .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
  677. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  678. },
  679. {
  680. .label = "fan4:green",
  681. .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
  682. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  683. },
  684. {
  685. .label = "fan4:red",
  686. .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
  687. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  688. },
  689. };
  690. static struct mlxreg_core_platform_data mlxplat_default_led_data = {
  691. .data = mlxplat_mlxcpld_default_led_data,
  692. .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_led_data),
  693. };
  694. /* Platform led MSN21xx system family data */
  695. static struct mlxreg_core_data mlxplat_mlxcpld_msn21xx_led_data[] = {
  696. {
  697. .label = "status:green",
  698. .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
  699. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  700. },
  701. {
  702. .label = "status:red",
  703. .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
  704. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK
  705. },
  706. {
  707. .label = "fan:green",
  708. .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
  709. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  710. },
  711. {
  712. .label = "fan:red",
  713. .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
  714. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  715. },
  716. {
  717. .label = "psu1:green",
  718. .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
  719. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  720. },
  721. {
  722. .label = "psu1:red",
  723. .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
  724. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  725. },
  726. {
  727. .label = "psu2:green",
  728. .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
  729. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  730. },
  731. {
  732. .label = "psu2:red",
  733. .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
  734. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  735. },
  736. {
  737. .label = "uid:blue",
  738. .reg = MLXPLAT_CPLD_LPC_REG_LED5_OFFSET,
  739. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  740. },
  741. };
  742. static struct mlxreg_core_platform_data mlxplat_msn21xx_led_data = {
  743. .data = mlxplat_mlxcpld_msn21xx_led_data,
  744. .counter = ARRAY_SIZE(mlxplat_mlxcpld_msn21xx_led_data),
  745. };
  746. /* Platform led for default data for 200GbE systems */
  747. static struct mlxreg_core_data mlxplat_mlxcpld_default_ng_led_data[] = {
  748. {
  749. .label = "status:green",
  750. .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
  751. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  752. },
  753. {
  754. .label = "status:orange",
  755. .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
  756. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK
  757. },
  758. {
  759. .label = "psu:green",
  760. .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
  761. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  762. },
  763. {
  764. .label = "psu:orange",
  765. .reg = MLXPLAT_CPLD_LPC_REG_LED1_OFFSET,
  766. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  767. },
  768. {
  769. .label = "fan1:green",
  770. .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
  771. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  772. },
  773. {
  774. .label = "fan1:orange",
  775. .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
  776. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  777. },
  778. {
  779. .label = "fan2:green",
  780. .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
  781. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  782. },
  783. {
  784. .label = "fan2:orange",
  785. .reg = MLXPLAT_CPLD_LPC_REG_LED2_OFFSET,
  786. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  787. },
  788. {
  789. .label = "fan3:green",
  790. .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
  791. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  792. },
  793. {
  794. .label = "fan3:orange",
  795. .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
  796. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  797. },
  798. {
  799. .label = "fan4:green",
  800. .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
  801. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  802. },
  803. {
  804. .label = "fan4:orange",
  805. .reg = MLXPLAT_CPLD_LPC_REG_LED3_OFFSET,
  806. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  807. },
  808. {
  809. .label = "fan5:green",
  810. .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
  811. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  812. },
  813. {
  814. .label = "fan5:orange",
  815. .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
  816. .mask = MLXPLAT_CPLD_LED_LO_NIBBLE_MASK,
  817. },
  818. {
  819. .label = "fan6:green",
  820. .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
  821. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  822. },
  823. {
  824. .label = "fan6:orange",
  825. .reg = MLXPLAT_CPLD_LPC_REG_LED4_OFFSET,
  826. .mask = MLXPLAT_CPLD_LED_HI_NIBBLE_MASK,
  827. },
  828. };
  829. static struct mlxreg_core_platform_data mlxplat_default_ng_led_data = {
  830. .data = mlxplat_mlxcpld_default_ng_led_data,
  831. .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_ng_led_data),
  832. };
  833. /* Platform register access default */
  834. static struct mlxreg_core_data mlxplat_mlxcpld_default_regs_io_data[] = {
  835. {
  836. .label = "cpld1_version",
  837. .reg = MLXPLAT_CPLD_LPC_REG_CPLD1_VER_OFFSET,
  838. .bit = GENMASK(7, 0),
  839. .mode = 0444,
  840. },
  841. {
  842. .label = "cpld2_version",
  843. .reg = MLXPLAT_CPLD_LPC_REG_CPLD2_VER_OFFSET,
  844. .bit = GENMASK(7, 0),
  845. .mode = 0444,
  846. },
  847. {
  848. .label = "reset_long_pb",
  849. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  850. .mask = GENMASK(7, 0) & ~BIT(0),
  851. .mode = 0444,
  852. },
  853. {
  854. .label = "reset_short_pb",
  855. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  856. .mask = GENMASK(7, 0) & ~BIT(1),
  857. .mode = 0444,
  858. },
  859. {
  860. .label = "reset_aux_pwr_or_ref",
  861. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  862. .mask = GENMASK(7, 0) & ~BIT(2),
  863. .mode = 0444,
  864. },
  865. {
  866. .label = "reset_main_pwr_fail",
  867. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  868. .mask = GENMASK(7, 0) & ~BIT(3),
  869. .mode = 0444,
  870. },
  871. {
  872. .label = "reset_sw_reset",
  873. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  874. .mask = GENMASK(7, 0) & ~BIT(4),
  875. .mode = 0444,
  876. },
  877. {
  878. .label = "reset_fw_reset",
  879. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  880. .mask = GENMASK(7, 0) & ~BIT(5),
  881. .mode = 0444,
  882. },
  883. {
  884. .label = "reset_hotswap_or_wd",
  885. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  886. .mask = GENMASK(7, 0) & ~BIT(6),
  887. .mode = 0444,
  888. },
  889. {
  890. .label = "reset_asic_thermal",
  891. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  892. .mask = GENMASK(7, 0) & ~BIT(7),
  893. .mode = 0444,
  894. },
  895. {
  896. .label = "psu1_on",
  897. .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
  898. .mask = GENMASK(7, 0) & ~BIT(0),
  899. .mode = 0200,
  900. },
  901. {
  902. .label = "psu2_on",
  903. .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
  904. .mask = GENMASK(7, 0) & ~BIT(1),
  905. .mode = 0200,
  906. },
  907. {
  908. .label = "pwr_cycle",
  909. .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
  910. .mask = GENMASK(7, 0) & ~BIT(2),
  911. .mode = 0200,
  912. },
  913. {
  914. .label = "pwr_down",
  915. .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
  916. .mask = GENMASK(7, 0) & ~BIT(3),
  917. .mode = 0200,
  918. },
  919. {
  920. .label = "select_iio",
  921. .reg = MLXPLAT_CPLD_LPC_REG_GP2_OFFSET,
  922. .mask = GENMASK(7, 0) & ~BIT(6),
  923. .mode = 0644,
  924. },
  925. {
  926. .label = "asic_health",
  927. .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
  928. .mask = MLXPLAT_CPLD_ASIC_MASK,
  929. .bit = 1,
  930. .mode = 0444,
  931. },
  932. };
  933. static struct mlxreg_core_platform_data mlxplat_default_regs_io_data = {
  934. .data = mlxplat_mlxcpld_default_regs_io_data,
  935. .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_regs_io_data),
  936. };
  937. /* Platform register access MSN21xx, MSN201x, MSN274x systems families data */
  938. static struct mlxreg_core_data mlxplat_mlxcpld_msn21xx_regs_io_data[] = {
  939. {
  940. .label = "cpld1_version",
  941. .reg = MLXPLAT_CPLD_LPC_REG_CPLD1_VER_OFFSET,
  942. .bit = GENMASK(7, 0),
  943. .mode = 0444,
  944. },
  945. {
  946. .label = "cpld2_version",
  947. .reg = MLXPLAT_CPLD_LPC_REG_CPLD2_VER_OFFSET,
  948. .bit = GENMASK(7, 0),
  949. .mode = 0444,
  950. },
  951. {
  952. .label = "reset_long_pb",
  953. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  954. .mask = GENMASK(7, 0) & ~BIT(0),
  955. .mode = 0444,
  956. },
  957. {
  958. .label = "reset_short_pb",
  959. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  960. .mask = GENMASK(7, 0) & ~BIT(1),
  961. .mode = 0444,
  962. },
  963. {
  964. .label = "reset_aux_pwr_or_ref",
  965. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  966. .mask = GENMASK(7, 0) & ~BIT(2),
  967. .mode = 0444,
  968. },
  969. {
  970. .label = "reset_sw_reset",
  971. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  972. .mask = GENMASK(7, 0) & ~BIT(3),
  973. .mode = 0444,
  974. },
  975. {
  976. .label = "reset_main_pwr_fail",
  977. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  978. .mask = GENMASK(7, 0) & ~BIT(4),
  979. .mode = 0444,
  980. },
  981. {
  982. .label = "reset_asic_thermal",
  983. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  984. .mask = GENMASK(7, 0) & ~BIT(5),
  985. .mode = 0444,
  986. },
  987. {
  988. .label = "reset_hotswap_or_halt",
  989. .reg = MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET,
  990. .mask = GENMASK(7, 0) & ~BIT(6),
  991. .mode = 0444,
  992. },
  993. {
  994. .label = "psu1_on",
  995. .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
  996. .mask = GENMASK(7, 0) & ~BIT(0),
  997. .mode = 0200,
  998. },
  999. {
  1000. .label = "psu2_on",
  1001. .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
  1002. .mask = GENMASK(7, 0) & ~BIT(1),
  1003. .mode = 0200,
  1004. },
  1005. {
  1006. .label = "pwr_cycle",
  1007. .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
  1008. .mask = GENMASK(7, 0) & ~BIT(2),
  1009. .mode = 0200,
  1010. },
  1011. {
  1012. .label = "pwr_down",
  1013. .reg = MLXPLAT_CPLD_LPC_REG_GP1_OFFSET,
  1014. .mask = GENMASK(7, 0) & ~BIT(3),
  1015. .mode = 0200,
  1016. },
  1017. {
  1018. .label = "asic_health",
  1019. .reg = MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET,
  1020. .mask = MLXPLAT_CPLD_ASIC_MASK,
  1021. .bit = 1,
  1022. .mode = 0444,
  1023. },
  1024. };
  1025. static struct mlxreg_core_platform_data mlxplat_msn21xx_regs_io_data = {
  1026. .data = mlxplat_mlxcpld_msn21xx_regs_io_data,
  1027. .counter = ARRAY_SIZE(mlxplat_mlxcpld_msn21xx_regs_io_data),
  1028. };
  1029. /* Platform FAN default */
  1030. static struct mlxreg_core_data mlxplat_mlxcpld_default_fan_data[] = {
  1031. {
  1032. .label = "pwm1",
  1033. .reg = MLXPLAT_CPLD_LPC_REG_PWM1_OFFSET,
  1034. },
  1035. {
  1036. .label = "tacho1",
  1037. .reg = MLXPLAT_CPLD_LPC_REG_TACHO1_OFFSET,
  1038. .mask = GENMASK(7, 0),
  1039. },
  1040. {
  1041. .label = "tacho2",
  1042. .reg = MLXPLAT_CPLD_LPC_REG_TACHO2_OFFSET,
  1043. .mask = GENMASK(7, 0),
  1044. },
  1045. {
  1046. .label = "tacho3",
  1047. .reg = MLXPLAT_CPLD_LPC_REG_TACHO3_OFFSET,
  1048. .mask = GENMASK(7, 0),
  1049. },
  1050. {
  1051. .label = "tacho4",
  1052. .reg = MLXPLAT_CPLD_LPC_REG_TACHO4_OFFSET,
  1053. .mask = GENMASK(7, 0),
  1054. },
  1055. {
  1056. .label = "tacho5",
  1057. .reg = MLXPLAT_CPLD_LPC_REG_TACHO5_OFFSET,
  1058. .mask = GENMASK(7, 0),
  1059. },
  1060. {
  1061. .label = "tacho6",
  1062. .reg = MLXPLAT_CPLD_LPC_REG_TACHO6_OFFSET,
  1063. .mask = GENMASK(7, 0),
  1064. },
  1065. {
  1066. .label = "tacho7",
  1067. .reg = MLXPLAT_CPLD_LPC_REG_TACHO7_OFFSET,
  1068. .mask = GENMASK(7, 0),
  1069. },
  1070. {
  1071. .label = "tacho8",
  1072. .reg = MLXPLAT_CPLD_LPC_REG_TACHO8_OFFSET,
  1073. .mask = GENMASK(7, 0),
  1074. },
  1075. {
  1076. .label = "tacho9",
  1077. .reg = MLXPLAT_CPLD_LPC_REG_TACHO9_OFFSET,
  1078. .mask = GENMASK(7, 0),
  1079. },
  1080. {
  1081. .label = "tacho10",
  1082. .reg = MLXPLAT_CPLD_LPC_REG_TACHO10_OFFSET,
  1083. .mask = GENMASK(7, 0),
  1084. },
  1085. {
  1086. .label = "tacho11",
  1087. .reg = MLXPLAT_CPLD_LPC_REG_TACHO11_OFFSET,
  1088. .mask = GENMASK(7, 0),
  1089. },
  1090. {
  1091. .label = "tacho12",
  1092. .reg = MLXPLAT_CPLD_LPC_REG_TACHO12_OFFSET,
  1093. .mask = GENMASK(7, 0),
  1094. },
  1095. };
  1096. static struct mlxreg_core_platform_data mlxplat_default_fan_data = {
  1097. .data = mlxplat_mlxcpld_default_fan_data,
  1098. .counter = ARRAY_SIZE(mlxplat_mlxcpld_default_fan_data),
  1099. };
  1100. static bool mlxplat_mlxcpld_writeable_reg(struct device *dev, unsigned int reg)
  1101. {
  1102. switch (reg) {
  1103. case MLXPLAT_CPLD_LPC_REG_LED1_OFFSET:
  1104. case MLXPLAT_CPLD_LPC_REG_LED2_OFFSET:
  1105. case MLXPLAT_CPLD_LPC_REG_LED3_OFFSET:
  1106. case MLXPLAT_CPLD_LPC_REG_LED4_OFFSET:
  1107. case MLXPLAT_CPLD_LPC_REG_LED5_OFFSET:
  1108. case MLXPLAT_CPLD_LPC_REG_GP1_OFFSET:
  1109. case MLXPLAT_CPLD_LPC_REG_WP1_OFFSET:
  1110. case MLXPLAT_CPLD_LPC_REG_GP2_OFFSET:
  1111. case MLXPLAT_CPLD_LPC_REG_WP2_OFFSET:
  1112. case MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET:
  1113. case MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET:
  1114. case MLXPLAT_CPLD_LPC_REG_ASIC_EVENT_OFFSET:
  1115. case MLXPLAT_CPLD_LPC_REG_ASIC_MASK_OFFSET:
  1116. case MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET:
  1117. case MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET:
  1118. case MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET:
  1119. case MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET:
  1120. case MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET:
  1121. case MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET:
  1122. case MLXPLAT_CPLD_LPC_REG_PWM1_OFFSET:
  1123. case MLXPLAT_CPLD_LPC_REG_PWM_CONTROL_OFFSET:
  1124. return true;
  1125. }
  1126. return false;
  1127. }
  1128. static bool mlxplat_mlxcpld_readable_reg(struct device *dev, unsigned int reg)
  1129. {
  1130. switch (reg) {
  1131. case MLXPLAT_CPLD_LPC_REG_CPLD1_VER_OFFSET:
  1132. case MLXPLAT_CPLD_LPC_REG_CPLD2_VER_OFFSET:
  1133. case MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET:
  1134. case MLXPLAT_CPLD_LPC_REG_LED1_OFFSET:
  1135. case MLXPLAT_CPLD_LPC_REG_LED2_OFFSET:
  1136. case MLXPLAT_CPLD_LPC_REG_LED3_OFFSET:
  1137. case MLXPLAT_CPLD_LPC_REG_LED4_OFFSET:
  1138. case MLXPLAT_CPLD_LPC_REG_LED5_OFFSET:
  1139. case MLXPLAT_CPLD_LPC_REG_GP1_OFFSET:
  1140. case MLXPLAT_CPLD_LPC_REG_WP1_OFFSET:
  1141. case MLXPLAT_CPLD_LPC_REG_GP2_OFFSET:
  1142. case MLXPLAT_CPLD_LPC_REG_WP2_OFFSET:
  1143. case MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET:
  1144. case MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET:
  1145. case MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET:
  1146. case MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET:
  1147. case MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET:
  1148. case MLXPLAT_CPLD_LPC_REG_ASIC_EVENT_OFFSET:
  1149. case MLXPLAT_CPLD_LPC_REG_ASIC_MASK_OFFSET:
  1150. case MLXPLAT_CPLD_LPC_REG_PSU_OFFSET:
  1151. case MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET:
  1152. case MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET:
  1153. case MLXPLAT_CPLD_LPC_REG_PWR_OFFSET:
  1154. case MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET:
  1155. case MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET:
  1156. case MLXPLAT_CPLD_LPC_REG_FAN_OFFSET:
  1157. case MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET:
  1158. case MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET:
  1159. case MLXPLAT_CPLD_LPC_REG_PWM1_OFFSET:
  1160. case MLXPLAT_CPLD_LPC_REG_TACHO1_OFFSET:
  1161. case MLXPLAT_CPLD_LPC_REG_TACHO2_OFFSET:
  1162. case MLXPLAT_CPLD_LPC_REG_TACHO3_OFFSET:
  1163. case MLXPLAT_CPLD_LPC_REG_TACHO4_OFFSET:
  1164. case MLXPLAT_CPLD_LPC_REG_TACHO5_OFFSET:
  1165. case MLXPLAT_CPLD_LPC_REG_TACHO6_OFFSET:
  1166. case MLXPLAT_CPLD_LPC_REG_TACHO7_OFFSET:
  1167. case MLXPLAT_CPLD_LPC_REG_TACHO8_OFFSET:
  1168. case MLXPLAT_CPLD_LPC_REG_TACHO9_OFFSET:
  1169. case MLXPLAT_CPLD_LPC_REG_TACHO10_OFFSET:
  1170. case MLXPLAT_CPLD_LPC_REG_TACHO11_OFFSET:
  1171. case MLXPLAT_CPLD_LPC_REG_TACHO12_OFFSET:
  1172. case MLXPLAT_CPLD_LPC_REG_PWM_CONTROL_OFFSET:
  1173. return true;
  1174. }
  1175. return false;
  1176. }
  1177. static bool mlxplat_mlxcpld_volatile_reg(struct device *dev, unsigned int reg)
  1178. {
  1179. switch (reg) {
  1180. case MLXPLAT_CPLD_LPC_REG_CPLD1_VER_OFFSET:
  1181. case MLXPLAT_CPLD_LPC_REG_CPLD2_VER_OFFSET:
  1182. case MLXPLAT_CPLD_LPC_REG_RESET_CAUSE_OFFSET:
  1183. case MLXPLAT_CPLD_LPC_REG_LED1_OFFSET:
  1184. case MLXPLAT_CPLD_LPC_REG_LED2_OFFSET:
  1185. case MLXPLAT_CPLD_LPC_REG_LED3_OFFSET:
  1186. case MLXPLAT_CPLD_LPC_REG_LED4_OFFSET:
  1187. case MLXPLAT_CPLD_LPC_REG_LED5_OFFSET:
  1188. case MLXPLAT_CPLD_LPC_REG_GP1_OFFSET:
  1189. case MLXPLAT_CPLD_LPC_REG_GP2_OFFSET:
  1190. case MLXPLAT_CPLD_LPC_REG_AGGR_OFFSET:
  1191. case MLXPLAT_CPLD_LPC_REG_AGGR_MASK_OFFSET:
  1192. case MLXPLAT_CPLD_LPC_REG_AGGRLO_OFFSET:
  1193. case MLXPLAT_CPLD_LPC_REG_AGGRLO_MASK_OFFSET:
  1194. case MLXPLAT_CPLD_LPC_REG_ASIC_HEALTH_OFFSET:
  1195. case MLXPLAT_CPLD_LPC_REG_ASIC_EVENT_OFFSET:
  1196. case MLXPLAT_CPLD_LPC_REG_ASIC_MASK_OFFSET:
  1197. case MLXPLAT_CPLD_LPC_REG_PSU_OFFSET:
  1198. case MLXPLAT_CPLD_LPC_REG_PSU_EVENT_OFFSET:
  1199. case MLXPLAT_CPLD_LPC_REG_PSU_MASK_OFFSET:
  1200. case MLXPLAT_CPLD_LPC_REG_PWR_OFFSET:
  1201. case MLXPLAT_CPLD_LPC_REG_PWR_EVENT_OFFSET:
  1202. case MLXPLAT_CPLD_LPC_REG_PWR_MASK_OFFSET:
  1203. case MLXPLAT_CPLD_LPC_REG_FAN_OFFSET:
  1204. case MLXPLAT_CPLD_LPC_REG_FAN_EVENT_OFFSET:
  1205. case MLXPLAT_CPLD_LPC_REG_FAN_MASK_OFFSET:
  1206. case MLXPLAT_CPLD_LPC_REG_PWM1_OFFSET:
  1207. case MLXPLAT_CPLD_LPC_REG_TACHO1_OFFSET:
  1208. case MLXPLAT_CPLD_LPC_REG_TACHO2_OFFSET:
  1209. case MLXPLAT_CPLD_LPC_REG_TACHO3_OFFSET:
  1210. case MLXPLAT_CPLD_LPC_REG_TACHO4_OFFSET:
  1211. case MLXPLAT_CPLD_LPC_REG_TACHO5_OFFSET:
  1212. case MLXPLAT_CPLD_LPC_REG_TACHO6_OFFSET:
  1213. case MLXPLAT_CPLD_LPC_REG_TACHO7_OFFSET:
  1214. case MLXPLAT_CPLD_LPC_REG_TACHO8_OFFSET:
  1215. case MLXPLAT_CPLD_LPC_REG_TACHO9_OFFSET:
  1216. case MLXPLAT_CPLD_LPC_REG_TACHO10_OFFSET:
  1217. case MLXPLAT_CPLD_LPC_REG_TACHO11_OFFSET:
  1218. case MLXPLAT_CPLD_LPC_REG_TACHO12_OFFSET:
  1219. case MLXPLAT_CPLD_LPC_REG_PWM_CONTROL_OFFSET:
  1220. return true;
  1221. }
  1222. return false;
  1223. }
  1224. static const struct reg_default mlxplat_mlxcpld_regmap_default[] = {
  1225. { MLXPLAT_CPLD_LPC_REG_WP1_OFFSET, 0x00 },
  1226. { MLXPLAT_CPLD_LPC_REG_WP2_OFFSET, 0x00 },
  1227. { MLXPLAT_CPLD_LPC_REG_PWM_CONTROL_OFFSET, 0x00 },
  1228. };
  1229. struct mlxplat_mlxcpld_regmap_context {
  1230. void __iomem *base;
  1231. };
  1232. static struct mlxplat_mlxcpld_regmap_context mlxplat_mlxcpld_regmap_ctx;
  1233. static int
  1234. mlxplat_mlxcpld_reg_read(void *context, unsigned int reg, unsigned int *val)
  1235. {
  1236. struct mlxplat_mlxcpld_regmap_context *ctx = context;
  1237. *val = ioread8(ctx->base + reg);
  1238. return 0;
  1239. }
  1240. static int
  1241. mlxplat_mlxcpld_reg_write(void *context, unsigned int reg, unsigned int val)
  1242. {
  1243. struct mlxplat_mlxcpld_regmap_context *ctx = context;
  1244. iowrite8(val, ctx->base + reg);
  1245. return 0;
  1246. }
  1247. static const struct regmap_config mlxplat_mlxcpld_regmap_config = {
  1248. .reg_bits = 8,
  1249. .val_bits = 8,
  1250. .max_register = 255,
  1251. .cache_type = REGCACHE_FLAT,
  1252. .writeable_reg = mlxplat_mlxcpld_writeable_reg,
  1253. .readable_reg = mlxplat_mlxcpld_readable_reg,
  1254. .volatile_reg = mlxplat_mlxcpld_volatile_reg,
  1255. .reg_defaults = mlxplat_mlxcpld_regmap_default,
  1256. .num_reg_defaults = ARRAY_SIZE(mlxplat_mlxcpld_regmap_default),
  1257. .reg_read = mlxplat_mlxcpld_reg_read,
  1258. .reg_write = mlxplat_mlxcpld_reg_write,
  1259. };
  1260. static struct resource mlxplat_mlxcpld_resources[] = {
  1261. [0] = DEFINE_RES_IRQ_NAMED(17, "mlxreg-hotplug"),
  1262. };
  1263. static struct platform_device *mlxplat_dev;
  1264. static struct mlxreg_core_hotplug_platform_data *mlxplat_hotplug;
  1265. static struct mlxreg_core_platform_data *mlxplat_led;
  1266. static struct mlxreg_core_platform_data *mlxplat_regs_io;
  1267. static struct mlxreg_core_platform_data *mlxplat_fan;
  1268. static int __init mlxplat_dmi_default_matched(const struct dmi_system_id *dmi)
  1269. {
  1270. int i;
  1271. for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
  1272. mlxplat_mux_data[i].values = mlxplat_default_channels[i];
  1273. mlxplat_mux_data[i].n_values =
  1274. ARRAY_SIZE(mlxplat_default_channels[i]);
  1275. }
  1276. mlxplat_hotplug = &mlxplat_mlxcpld_default_data;
  1277. mlxplat_hotplug->deferred_nr =
  1278. mlxplat_default_channels[i - 1][MLXPLAT_CPLD_GRP_CHNL_NUM - 1];
  1279. mlxplat_led = &mlxplat_default_led_data;
  1280. mlxplat_regs_io = &mlxplat_default_regs_io_data;
  1281. return 1;
  1282. };
  1283. static int __init mlxplat_dmi_msn21xx_matched(const struct dmi_system_id *dmi)
  1284. {
  1285. int i;
  1286. for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
  1287. mlxplat_mux_data[i].values = mlxplat_msn21xx_channels;
  1288. mlxplat_mux_data[i].n_values =
  1289. ARRAY_SIZE(mlxplat_msn21xx_channels);
  1290. }
  1291. mlxplat_hotplug = &mlxplat_mlxcpld_msn21xx_data;
  1292. mlxplat_hotplug->deferred_nr =
  1293. mlxplat_msn21xx_channels[MLXPLAT_CPLD_GRP_CHNL_NUM - 1];
  1294. mlxplat_led = &mlxplat_msn21xx_led_data;
  1295. mlxplat_regs_io = &mlxplat_msn21xx_regs_io_data;
  1296. return 1;
  1297. };
  1298. static int __init mlxplat_dmi_msn274x_matched(const struct dmi_system_id *dmi)
  1299. {
  1300. int i;
  1301. for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
  1302. mlxplat_mux_data[i].values = mlxplat_msn21xx_channels;
  1303. mlxplat_mux_data[i].n_values =
  1304. ARRAY_SIZE(mlxplat_msn21xx_channels);
  1305. }
  1306. mlxplat_hotplug = &mlxplat_mlxcpld_msn274x_data;
  1307. mlxplat_hotplug->deferred_nr =
  1308. mlxplat_msn21xx_channels[MLXPLAT_CPLD_GRP_CHNL_NUM - 1];
  1309. mlxplat_led = &mlxplat_default_led_data;
  1310. mlxplat_regs_io = &mlxplat_msn21xx_regs_io_data;
  1311. return 1;
  1312. };
  1313. static int __init mlxplat_dmi_msn201x_matched(const struct dmi_system_id *dmi)
  1314. {
  1315. int i;
  1316. for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
  1317. mlxplat_mux_data[i].values = mlxplat_msn21xx_channels;
  1318. mlxplat_mux_data[i].n_values =
  1319. ARRAY_SIZE(mlxplat_msn21xx_channels);
  1320. }
  1321. mlxplat_hotplug = &mlxplat_mlxcpld_msn201x_data;
  1322. mlxplat_hotplug->deferred_nr =
  1323. mlxplat_default_channels[i - 1][MLXPLAT_CPLD_GRP_CHNL_NUM - 1];
  1324. mlxplat_led = &mlxplat_msn21xx_led_data;
  1325. mlxplat_regs_io = &mlxplat_msn21xx_regs_io_data;
  1326. return 1;
  1327. };
  1328. static int __init mlxplat_dmi_qmb7xx_matched(const struct dmi_system_id *dmi)
  1329. {
  1330. int i;
  1331. for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
  1332. mlxplat_mux_data[i].values = mlxplat_msn21xx_channels;
  1333. mlxplat_mux_data[i].n_values =
  1334. ARRAY_SIZE(mlxplat_msn21xx_channels);
  1335. }
  1336. mlxplat_hotplug = &mlxplat_mlxcpld_default_ng_data;
  1337. mlxplat_hotplug->deferred_nr =
  1338. mlxplat_msn21xx_channels[MLXPLAT_CPLD_GRP_CHNL_NUM - 1];
  1339. mlxplat_led = &mlxplat_default_ng_led_data;
  1340. mlxplat_fan = &mlxplat_default_fan_data;
  1341. return 1;
  1342. };
  1343. static const struct dmi_system_id mlxplat_dmi_table[] __initconst = {
  1344. {
  1345. .callback = mlxplat_dmi_msn274x_matched,
  1346. .matches = {
  1347. DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
  1348. DMI_MATCH(DMI_PRODUCT_NAME, "MSN274"),
  1349. },
  1350. },
  1351. {
  1352. .callback = mlxplat_dmi_default_matched,
  1353. .matches = {
  1354. DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
  1355. DMI_MATCH(DMI_PRODUCT_NAME, "MSN24"),
  1356. },
  1357. },
  1358. {
  1359. .callback = mlxplat_dmi_default_matched,
  1360. .matches = {
  1361. DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
  1362. DMI_MATCH(DMI_PRODUCT_NAME, "MSN27"),
  1363. },
  1364. },
  1365. {
  1366. .callback = mlxplat_dmi_default_matched,
  1367. .matches = {
  1368. DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
  1369. DMI_MATCH(DMI_PRODUCT_NAME, "MSB"),
  1370. },
  1371. },
  1372. {
  1373. .callback = mlxplat_dmi_default_matched,
  1374. .matches = {
  1375. DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
  1376. DMI_MATCH(DMI_PRODUCT_NAME, "MSX"),
  1377. },
  1378. },
  1379. {
  1380. .callback = mlxplat_dmi_msn21xx_matched,
  1381. .matches = {
  1382. DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
  1383. DMI_MATCH(DMI_PRODUCT_NAME, "MSN21"),
  1384. },
  1385. },
  1386. {
  1387. .callback = mlxplat_dmi_msn201x_matched,
  1388. .matches = {
  1389. DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
  1390. DMI_MATCH(DMI_PRODUCT_NAME, "MSN201"),
  1391. },
  1392. },
  1393. {
  1394. .callback = mlxplat_dmi_qmb7xx_matched,
  1395. .matches = {
  1396. DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
  1397. DMI_MATCH(DMI_PRODUCT_NAME, "QMB7"),
  1398. },
  1399. },
  1400. {
  1401. .callback = mlxplat_dmi_qmb7xx_matched,
  1402. .matches = {
  1403. DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
  1404. DMI_MATCH(DMI_PRODUCT_NAME, "SN37"),
  1405. },
  1406. },
  1407. {
  1408. .callback = mlxplat_dmi_qmb7xx_matched,
  1409. .matches = {
  1410. DMI_MATCH(DMI_BOARD_VENDOR, "Mellanox Technologies"),
  1411. DMI_MATCH(DMI_PRODUCT_NAME, "SN34"),
  1412. },
  1413. },
  1414. {
  1415. .callback = mlxplat_dmi_default_matched,
  1416. .matches = {
  1417. DMI_MATCH(DMI_BOARD_NAME, "VMOD0001"),
  1418. },
  1419. },
  1420. {
  1421. .callback = mlxplat_dmi_msn21xx_matched,
  1422. .matches = {
  1423. DMI_MATCH(DMI_BOARD_NAME, "VMOD0002"),
  1424. },
  1425. },
  1426. {
  1427. .callback = mlxplat_dmi_msn274x_matched,
  1428. .matches = {
  1429. DMI_MATCH(DMI_BOARD_NAME, "VMOD0003"),
  1430. },
  1431. },
  1432. {
  1433. .callback = mlxplat_dmi_msn201x_matched,
  1434. .matches = {
  1435. DMI_MATCH(DMI_BOARD_NAME, "VMOD0004"),
  1436. },
  1437. },
  1438. {
  1439. .callback = mlxplat_dmi_qmb7xx_matched,
  1440. .matches = {
  1441. DMI_MATCH(DMI_BOARD_NAME, "VMOD0005"),
  1442. },
  1443. },
  1444. { }
  1445. };
  1446. MODULE_DEVICE_TABLE(dmi, mlxplat_dmi_table);
  1447. static int mlxplat_mlxcpld_verify_bus_topology(int *nr)
  1448. {
  1449. struct i2c_adapter *search_adap;
  1450. int shift, i;
  1451. /* Scan adapters from expected id to verify it is free. */
  1452. *nr = MLXPLAT_CPLD_PHYS_ADAPTER_DEF_NR;
  1453. for (i = MLXPLAT_CPLD_PHYS_ADAPTER_DEF_NR; i <
  1454. MLXPLAT_CPLD_MAX_PHYS_ADAPTER_NUM; i++) {
  1455. search_adap = i2c_get_adapter(i);
  1456. if (search_adap) {
  1457. i2c_put_adapter(search_adap);
  1458. continue;
  1459. }
  1460. /* Return if expected parent adapter is free. */
  1461. if (i == MLXPLAT_CPLD_PHYS_ADAPTER_DEF_NR)
  1462. return 0;
  1463. break;
  1464. }
  1465. /* Return with error if free id for adapter is not found. */
  1466. if (i == MLXPLAT_CPLD_MAX_PHYS_ADAPTER_NUM)
  1467. return -ENODEV;
  1468. /* Shift adapter ids, since expected parent adapter is not free. */
  1469. *nr = i;
  1470. for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
  1471. shift = *nr - mlxplat_mux_data[i].parent;
  1472. mlxplat_mux_data[i].parent = *nr;
  1473. mlxplat_mux_data[i].base_nr += shift;
  1474. if (shift > 0)
  1475. mlxplat_hotplug->shift_nr = shift;
  1476. }
  1477. return 0;
  1478. }
  1479. static int __init mlxplat_init(void)
  1480. {
  1481. struct mlxplat_priv *priv;
  1482. int i, j, nr, err;
  1483. if (!dmi_check_system(mlxplat_dmi_table))
  1484. return -ENODEV;
  1485. mlxplat_dev = platform_device_register_simple(MLX_PLAT_DEVICE_NAME, -1,
  1486. mlxplat_lpc_resources,
  1487. ARRAY_SIZE(mlxplat_lpc_resources));
  1488. if (IS_ERR(mlxplat_dev))
  1489. return PTR_ERR(mlxplat_dev);
  1490. priv = devm_kzalloc(&mlxplat_dev->dev, sizeof(struct mlxplat_priv),
  1491. GFP_KERNEL);
  1492. if (!priv) {
  1493. err = -ENOMEM;
  1494. goto fail_alloc;
  1495. }
  1496. platform_set_drvdata(mlxplat_dev, priv);
  1497. err = mlxplat_mlxcpld_verify_bus_topology(&nr);
  1498. if (nr < 0)
  1499. goto fail_alloc;
  1500. nr = (nr == MLXPLAT_CPLD_MAX_PHYS_ADAPTER_NUM) ? -1 : nr;
  1501. priv->pdev_i2c = platform_device_register_simple("i2c_mlxcpld", nr,
  1502. NULL, 0);
  1503. if (IS_ERR(priv->pdev_i2c)) {
  1504. err = PTR_ERR(priv->pdev_i2c);
  1505. goto fail_alloc;
  1506. }
  1507. for (i = 0; i < ARRAY_SIZE(mlxplat_mux_data); i++) {
  1508. priv->pdev_mux[i] = platform_device_register_resndata(
  1509. &priv->pdev_i2c->dev,
  1510. "i2c-mux-reg", i, NULL,
  1511. 0, &mlxplat_mux_data[i],
  1512. sizeof(mlxplat_mux_data[i]));
  1513. if (IS_ERR(priv->pdev_mux[i])) {
  1514. err = PTR_ERR(priv->pdev_mux[i]);
  1515. goto fail_platform_mux_register;
  1516. }
  1517. }
  1518. mlxplat_mlxcpld_regmap_ctx.base = devm_ioport_map(&mlxplat_dev->dev,
  1519. mlxplat_lpc_resources[1].start, 1);
  1520. if (!mlxplat_mlxcpld_regmap_ctx.base) {
  1521. err = -ENOMEM;
  1522. goto fail_platform_mux_register;
  1523. }
  1524. mlxplat_hotplug->regmap = devm_regmap_init(&mlxplat_dev->dev, NULL,
  1525. &mlxplat_mlxcpld_regmap_ctx,
  1526. &mlxplat_mlxcpld_regmap_config);
  1527. if (IS_ERR(mlxplat_hotplug->regmap)) {
  1528. err = PTR_ERR(mlxplat_hotplug->regmap);
  1529. goto fail_platform_mux_register;
  1530. }
  1531. priv->pdev_hotplug = platform_device_register_resndata(
  1532. &mlxplat_dev->dev, "mlxreg-hotplug",
  1533. PLATFORM_DEVID_NONE,
  1534. mlxplat_mlxcpld_resources,
  1535. ARRAY_SIZE(mlxplat_mlxcpld_resources),
  1536. mlxplat_hotplug, sizeof(*mlxplat_hotplug));
  1537. if (IS_ERR(priv->pdev_hotplug)) {
  1538. err = PTR_ERR(priv->pdev_hotplug);
  1539. goto fail_platform_mux_register;
  1540. }
  1541. /* Set default registers. */
  1542. for (j = 0; j < mlxplat_mlxcpld_regmap_config.num_reg_defaults; j++) {
  1543. err = regmap_write(mlxplat_hotplug->regmap,
  1544. mlxplat_mlxcpld_regmap_default[j].reg,
  1545. mlxplat_mlxcpld_regmap_default[j].def);
  1546. if (err)
  1547. goto fail_platform_mux_register;
  1548. }
  1549. /* Add LED driver. */
  1550. mlxplat_led->regmap = mlxplat_hotplug->regmap;
  1551. priv->pdev_led = platform_device_register_resndata(
  1552. &mlxplat_dev->dev, "leds-mlxreg",
  1553. PLATFORM_DEVID_NONE, NULL, 0,
  1554. mlxplat_led, sizeof(*mlxplat_led));
  1555. if (IS_ERR(priv->pdev_led)) {
  1556. err = PTR_ERR(priv->pdev_led);
  1557. goto fail_platform_hotplug_register;
  1558. }
  1559. /* Add registers io access driver. */
  1560. if (mlxplat_regs_io) {
  1561. mlxplat_regs_io->regmap = mlxplat_hotplug->regmap;
  1562. priv->pdev_io_regs = platform_device_register_resndata(
  1563. &mlxplat_dev->dev, "mlxreg-io",
  1564. PLATFORM_DEVID_NONE, NULL, 0,
  1565. mlxplat_regs_io,
  1566. sizeof(*mlxplat_regs_io));
  1567. if (IS_ERR(priv->pdev_io_regs)) {
  1568. err = PTR_ERR(priv->pdev_io_regs);
  1569. goto fail_platform_led_register;
  1570. }
  1571. }
  1572. /* Add FAN driver. */
  1573. if (mlxplat_fan) {
  1574. mlxplat_fan->regmap = mlxplat_hotplug->regmap;
  1575. priv->pdev_fan = platform_device_register_resndata(
  1576. &mlxplat_dev->dev, "mlxreg-fan",
  1577. PLATFORM_DEVID_NONE, NULL, 0,
  1578. mlxplat_fan,
  1579. sizeof(*mlxplat_fan));
  1580. if (IS_ERR(priv->pdev_fan)) {
  1581. err = PTR_ERR(priv->pdev_fan);
  1582. goto fail_platform_io_regs_register;
  1583. }
  1584. }
  1585. /* Sync registers with hardware. */
  1586. regcache_mark_dirty(mlxplat_hotplug->regmap);
  1587. err = regcache_sync(mlxplat_hotplug->regmap);
  1588. if (err)
  1589. goto fail_platform_fan_register;
  1590. return 0;
  1591. fail_platform_fan_register:
  1592. if (mlxplat_fan)
  1593. platform_device_unregister(priv->pdev_fan);
  1594. fail_platform_io_regs_register:
  1595. if (mlxplat_regs_io)
  1596. platform_device_unregister(priv->pdev_io_regs);
  1597. fail_platform_led_register:
  1598. platform_device_unregister(priv->pdev_led);
  1599. fail_platform_hotplug_register:
  1600. platform_device_unregister(priv->pdev_hotplug);
  1601. fail_platform_mux_register:
  1602. while (--i >= 0)
  1603. platform_device_unregister(priv->pdev_mux[i]);
  1604. platform_device_unregister(priv->pdev_i2c);
  1605. fail_alloc:
  1606. platform_device_unregister(mlxplat_dev);
  1607. return err;
  1608. }
  1609. module_init(mlxplat_init);
  1610. static void __exit mlxplat_exit(void)
  1611. {
  1612. struct mlxplat_priv *priv = platform_get_drvdata(mlxplat_dev);
  1613. int i;
  1614. if (priv->pdev_fan)
  1615. platform_device_unregister(priv->pdev_fan);
  1616. if (priv->pdev_io_regs)
  1617. platform_device_unregister(priv->pdev_io_regs);
  1618. platform_device_unregister(priv->pdev_led);
  1619. platform_device_unregister(priv->pdev_hotplug);
  1620. for (i = ARRAY_SIZE(mlxplat_mux_data) - 1; i >= 0 ; i--)
  1621. platform_device_unregister(priv->pdev_mux[i]);
  1622. platform_device_unregister(priv->pdev_i2c);
  1623. platform_device_unregister(mlxplat_dev);
  1624. }
  1625. module_exit(mlxplat_exit);
  1626. MODULE_AUTHOR("Vadim Pasternak (vadimp@mellanox.com)");
  1627. MODULE_DESCRIPTION("Mellanox platform driver");
  1628. MODULE_LICENSE("Dual BSD/GPL");