14b7f9568SRashmica Gupta /*
24b7f9568SRashmica Gupta * ASPEED GPIO Controller
34b7f9568SRashmica Gupta *
44b7f9568SRashmica Gupta * Copyright (C) 2017-2019 IBM Corp.
54b7f9568SRashmica Gupta *
64b7f9568SRashmica Gupta * SPDX-License-Identifier: GPL-2.0-or-later
74b7f9568SRashmica Gupta */
84b7f9568SRashmica Gupta
94b7f9568SRashmica Gupta #include "qemu/osdep.h"
104b7f9568SRashmica Gupta #include "qemu/host-utils.h"
114b7f9568SRashmica Gupta #include "qemu/log.h"
124b7f9568SRashmica Gupta #include "hw/gpio/aspeed_gpio.h"
1358519090SPhilippe Mathieu-Daudé #include "hw/misc/aspeed_scu.h"
144b7f9568SRashmica Gupta #include "qapi/error.h"
154b7f9568SRashmica Gupta #include "qapi/visitor.h"
164b7f9568SRashmica Gupta #include "hw/irq.h"
174b7f9568SRashmica Gupta #include "migration/vmstate.h"
187b1d21a8SJamin Lin #include "trace.h"
19247c0029SJamin Lin #include "hw/registerfields.h"
204b7f9568SRashmica Gupta
214b7f9568SRashmica Gupta #define GPIOS_PER_GROUP 8
224b7f9568SRashmica Gupta
234b7f9568SRashmica Gupta /* GPIO Source Types */
244b7f9568SRashmica Gupta #define ASPEED_CMD_SRC_MASK 0x01010101
254b7f9568SRashmica Gupta #define ASPEED_SOURCE_ARM 0
264b7f9568SRashmica Gupta #define ASPEED_SOURCE_LPC 1
274b7f9568SRashmica Gupta #define ASPEED_SOURCE_COPROCESSOR 2
284b7f9568SRashmica Gupta #define ASPEED_SOURCE_RESERVED 3
294b7f9568SRashmica Gupta
304b7f9568SRashmica Gupta /* GPIO Interrupt Triggers */
314b7f9568SRashmica Gupta /*
324b7f9568SRashmica Gupta * For each set of gpios there are three sensitivity registers that control
334b7f9568SRashmica Gupta * the interrupt trigger mode.
344b7f9568SRashmica Gupta *
354b7f9568SRashmica Gupta * | 2 | 1 | 0 | trigger mode
364b7f9568SRashmica Gupta * -----------------------------
374b7f9568SRashmica Gupta * | 0 | 0 | 0 | falling-edge
384b7f9568SRashmica Gupta * | 0 | 0 | 1 | rising-edge
394b7f9568SRashmica Gupta * | 0 | 1 | 0 | level-low
404b7f9568SRashmica Gupta * | 0 | 1 | 1 | level-high
414b7f9568SRashmica Gupta * | 1 | X | X | dual-edge
424b7f9568SRashmica Gupta */
434b7f9568SRashmica Gupta #define ASPEED_FALLING_EDGE 0
444b7f9568SRashmica Gupta #define ASPEED_RISING_EDGE 1
454b7f9568SRashmica Gupta #define ASPEED_LEVEL_LOW 2
464b7f9568SRashmica Gupta #define ASPEED_LEVEL_HIGH 3
474b7f9568SRashmica Gupta #define ASPEED_DUAL_EDGE 4
484b7f9568SRashmica Gupta
494b7f9568SRashmica Gupta /* GPIO Register Address Offsets */
504b7f9568SRashmica Gupta #define GPIO_ABCD_DATA_VALUE (0x000 >> 2)
514b7f9568SRashmica Gupta #define GPIO_ABCD_DIRECTION (0x004 >> 2)
524b7f9568SRashmica Gupta #define GPIO_ABCD_INT_ENABLE (0x008 >> 2)
534b7f9568SRashmica Gupta #define GPIO_ABCD_INT_SENS_0 (0x00C >> 2)
544b7f9568SRashmica Gupta #define GPIO_ABCD_INT_SENS_1 (0x010 >> 2)
554b7f9568SRashmica Gupta #define GPIO_ABCD_INT_SENS_2 (0x014 >> 2)
564b7f9568SRashmica Gupta #define GPIO_ABCD_INT_STATUS (0x018 >> 2)
574b7f9568SRashmica Gupta #define GPIO_ABCD_RESET_TOLERANT (0x01C >> 2)
584b7f9568SRashmica Gupta #define GPIO_EFGH_DATA_VALUE (0x020 >> 2)
594b7f9568SRashmica Gupta #define GPIO_EFGH_DIRECTION (0x024 >> 2)
604b7f9568SRashmica Gupta #define GPIO_EFGH_INT_ENABLE (0x028 >> 2)
614b7f9568SRashmica Gupta #define GPIO_EFGH_INT_SENS_0 (0x02C >> 2)
624b7f9568SRashmica Gupta #define GPIO_EFGH_INT_SENS_1 (0x030 >> 2)
634b7f9568SRashmica Gupta #define GPIO_EFGH_INT_SENS_2 (0x034 >> 2)
644b7f9568SRashmica Gupta #define GPIO_EFGH_INT_STATUS (0x038 >> 2)
654b7f9568SRashmica Gupta #define GPIO_EFGH_RESET_TOLERANT (0x03C >> 2)
664b7f9568SRashmica Gupta #define GPIO_ABCD_DEBOUNCE_1 (0x040 >> 2)
674b7f9568SRashmica Gupta #define GPIO_ABCD_DEBOUNCE_2 (0x044 >> 2)
684b7f9568SRashmica Gupta #define GPIO_EFGH_DEBOUNCE_1 (0x048 >> 2)
694b7f9568SRashmica Gupta #define GPIO_EFGH_DEBOUNCE_2 (0x04C >> 2)
704b7f9568SRashmica Gupta #define GPIO_DEBOUNCE_TIME_1 (0x050 >> 2)
714b7f9568SRashmica Gupta #define GPIO_DEBOUNCE_TIME_2 (0x054 >> 2)
724b7f9568SRashmica Gupta #define GPIO_DEBOUNCE_TIME_3 (0x058 >> 2)
734b7f9568SRashmica Gupta #define GPIO_ABCD_COMMAND_SRC_0 (0x060 >> 2)
744b7f9568SRashmica Gupta #define GPIO_ABCD_COMMAND_SRC_1 (0x064 >> 2)
754b7f9568SRashmica Gupta #define GPIO_EFGH_COMMAND_SRC_0 (0x068 >> 2)
764b7f9568SRashmica Gupta #define GPIO_EFGH_COMMAND_SRC_1 (0x06C >> 2)
774b7f9568SRashmica Gupta #define GPIO_IJKL_DATA_VALUE (0x070 >> 2)
784b7f9568SRashmica Gupta #define GPIO_IJKL_DIRECTION (0x074 >> 2)
794b7f9568SRashmica Gupta #define GPIO_MNOP_DATA_VALUE (0x078 >> 2)
804b7f9568SRashmica Gupta #define GPIO_MNOP_DIRECTION (0x07C >> 2)
814b7f9568SRashmica Gupta #define GPIO_QRST_DATA_VALUE (0x080 >> 2)
824b7f9568SRashmica Gupta #define GPIO_QRST_DIRECTION (0x084 >> 2)
834b7f9568SRashmica Gupta #define GPIO_UVWX_DATA_VALUE (0x088 >> 2)
844b7f9568SRashmica Gupta #define GPIO_UVWX_DIRECTION (0x08C >> 2)
854b7f9568SRashmica Gupta #define GPIO_IJKL_COMMAND_SRC_0 (0x090 >> 2)
864b7f9568SRashmica Gupta #define GPIO_IJKL_COMMAND_SRC_1 (0x094 >> 2)
874b7f9568SRashmica Gupta #define GPIO_IJKL_INT_ENABLE (0x098 >> 2)
884b7f9568SRashmica Gupta #define GPIO_IJKL_INT_SENS_0 (0x09C >> 2)
894b7f9568SRashmica Gupta #define GPIO_IJKL_INT_SENS_1 (0x0A0 >> 2)
904b7f9568SRashmica Gupta #define GPIO_IJKL_INT_SENS_2 (0x0A4 >> 2)
914b7f9568SRashmica Gupta #define GPIO_IJKL_INT_STATUS (0x0A8 >> 2)
924b7f9568SRashmica Gupta #define GPIO_IJKL_RESET_TOLERANT (0x0AC >> 2)
934b7f9568SRashmica Gupta #define GPIO_IJKL_DEBOUNCE_1 (0x0B0 >> 2)
944b7f9568SRashmica Gupta #define GPIO_IJKL_DEBOUNCE_2 (0x0B4 >> 2)
954b7f9568SRashmica Gupta #define GPIO_IJKL_INPUT_MASK (0x0B8 >> 2)
964b7f9568SRashmica Gupta #define GPIO_ABCD_DATA_READ (0x0C0 >> 2)
974b7f9568SRashmica Gupta #define GPIO_EFGH_DATA_READ (0x0C4 >> 2)
984b7f9568SRashmica Gupta #define GPIO_IJKL_DATA_READ (0x0C8 >> 2)
994b7f9568SRashmica Gupta #define GPIO_MNOP_DATA_READ (0x0CC >> 2)
1004b7f9568SRashmica Gupta #define GPIO_QRST_DATA_READ (0x0D0 >> 2)
1014b7f9568SRashmica Gupta #define GPIO_UVWX_DATA_READ (0x0D4 >> 2)
1024b7f9568SRashmica Gupta #define GPIO_YZAAAB_DATA_READ (0x0D8 >> 2)
1034b7f9568SRashmica Gupta #define GPIO_AC_DATA_READ (0x0DC >> 2)
1044b7f9568SRashmica Gupta #define GPIO_MNOP_COMMAND_SRC_0 (0x0E0 >> 2)
1054b7f9568SRashmica Gupta #define GPIO_MNOP_COMMAND_SRC_1 (0x0E4 >> 2)
1064b7f9568SRashmica Gupta #define GPIO_MNOP_INT_ENABLE (0x0E8 >> 2)
1074b7f9568SRashmica Gupta #define GPIO_MNOP_INT_SENS_0 (0x0EC >> 2)
1084b7f9568SRashmica Gupta #define GPIO_MNOP_INT_SENS_1 (0x0F0 >> 2)
1094b7f9568SRashmica Gupta #define GPIO_MNOP_INT_SENS_2 (0x0F4 >> 2)
1104b7f9568SRashmica Gupta #define GPIO_MNOP_INT_STATUS (0x0F8 >> 2)
1114b7f9568SRashmica Gupta #define GPIO_MNOP_RESET_TOLERANT (0x0FC >> 2)
1124b7f9568SRashmica Gupta #define GPIO_MNOP_DEBOUNCE_1 (0x100 >> 2)
1134b7f9568SRashmica Gupta #define GPIO_MNOP_DEBOUNCE_2 (0x104 >> 2)
1144b7f9568SRashmica Gupta #define GPIO_MNOP_INPUT_MASK (0x108 >> 2)
1154b7f9568SRashmica Gupta #define GPIO_QRST_COMMAND_SRC_0 (0x110 >> 2)
1164b7f9568SRashmica Gupta #define GPIO_QRST_COMMAND_SRC_1 (0x114 >> 2)
1174b7f9568SRashmica Gupta #define GPIO_QRST_INT_ENABLE (0x118 >> 2)
1184b7f9568SRashmica Gupta #define GPIO_QRST_INT_SENS_0 (0x11C >> 2)
1194b7f9568SRashmica Gupta #define GPIO_QRST_INT_SENS_1 (0x120 >> 2)
1204b7f9568SRashmica Gupta #define GPIO_QRST_INT_SENS_2 (0x124 >> 2)
1214b7f9568SRashmica Gupta #define GPIO_QRST_INT_STATUS (0x128 >> 2)
1224b7f9568SRashmica Gupta #define GPIO_QRST_RESET_TOLERANT (0x12C >> 2)
1234b7f9568SRashmica Gupta #define GPIO_QRST_DEBOUNCE_1 (0x130 >> 2)
1244b7f9568SRashmica Gupta #define GPIO_QRST_DEBOUNCE_2 (0x134 >> 2)
1254b7f9568SRashmica Gupta #define GPIO_QRST_INPUT_MASK (0x138 >> 2)
1264b7f9568SRashmica Gupta #define GPIO_UVWX_COMMAND_SRC_0 (0x140 >> 2)
1274b7f9568SRashmica Gupta #define GPIO_UVWX_COMMAND_SRC_1 (0x144 >> 2)
1284b7f9568SRashmica Gupta #define GPIO_UVWX_INT_ENABLE (0x148 >> 2)
1294b7f9568SRashmica Gupta #define GPIO_UVWX_INT_SENS_0 (0x14C >> 2)
1304b7f9568SRashmica Gupta #define GPIO_UVWX_INT_SENS_1 (0x150 >> 2)
1314b7f9568SRashmica Gupta #define GPIO_UVWX_INT_SENS_2 (0x154 >> 2)
1324b7f9568SRashmica Gupta #define GPIO_UVWX_INT_STATUS (0x158 >> 2)
1334b7f9568SRashmica Gupta #define GPIO_UVWX_RESET_TOLERANT (0x15C >> 2)
1344b7f9568SRashmica Gupta #define GPIO_UVWX_DEBOUNCE_1 (0x160 >> 2)
1354b7f9568SRashmica Gupta #define GPIO_UVWX_DEBOUNCE_2 (0x164 >> 2)
1364b7f9568SRashmica Gupta #define GPIO_UVWX_INPUT_MASK (0x168 >> 2)
1374b7f9568SRashmica Gupta #define GPIO_YZAAAB_COMMAND_SRC_0 (0x170 >> 2)
1384b7f9568SRashmica Gupta #define GPIO_YZAAAB_COMMAND_SRC_1 (0x174 >> 2)
1394b7f9568SRashmica Gupta #define GPIO_YZAAAB_INT_ENABLE (0x178 >> 2)
1404b7f9568SRashmica Gupta #define GPIO_YZAAAB_INT_SENS_0 (0x17C >> 2)
1414b7f9568SRashmica Gupta #define GPIO_YZAAAB_INT_SENS_1 (0x180 >> 2)
1424b7f9568SRashmica Gupta #define GPIO_YZAAAB_INT_SENS_2 (0x184 >> 2)
1434b7f9568SRashmica Gupta #define GPIO_YZAAAB_INT_STATUS (0x188 >> 2)
1444b7f9568SRashmica Gupta #define GPIO_YZAAAB_RESET_TOLERANT (0x18C >> 2)
1454b7f9568SRashmica Gupta #define GPIO_YZAAAB_DEBOUNCE_1 (0x190 >> 2)
1464b7f9568SRashmica Gupta #define GPIO_YZAAAB_DEBOUNCE_2 (0x194 >> 2)
1474b7f9568SRashmica Gupta #define GPIO_YZAAAB_INPUT_MASK (0x198 >> 2)
1484b7f9568SRashmica Gupta #define GPIO_AC_COMMAND_SRC_0 (0x1A0 >> 2)
1494b7f9568SRashmica Gupta #define GPIO_AC_COMMAND_SRC_1 (0x1A4 >> 2)
1504b7f9568SRashmica Gupta #define GPIO_AC_INT_ENABLE (0x1A8 >> 2)
1514b7f9568SRashmica Gupta #define GPIO_AC_INT_SENS_0 (0x1AC >> 2)
1524b7f9568SRashmica Gupta #define GPIO_AC_INT_SENS_1 (0x1B0 >> 2)
1534b7f9568SRashmica Gupta #define GPIO_AC_INT_SENS_2 (0x1B4 >> 2)
1544b7f9568SRashmica Gupta #define GPIO_AC_INT_STATUS (0x1B8 >> 2)
1554b7f9568SRashmica Gupta #define GPIO_AC_RESET_TOLERANT (0x1BC >> 2)
1564b7f9568SRashmica Gupta #define GPIO_AC_DEBOUNCE_1 (0x1C0 >> 2)
1574b7f9568SRashmica Gupta #define GPIO_AC_DEBOUNCE_2 (0x1C4 >> 2)
1584b7f9568SRashmica Gupta #define GPIO_AC_INPUT_MASK (0x1C8 >> 2)
1594b7f9568SRashmica Gupta #define GPIO_ABCD_INPUT_MASK (0x1D0 >> 2)
1604b7f9568SRashmica Gupta #define GPIO_EFGH_INPUT_MASK (0x1D4 >> 2)
1614b7f9568SRashmica Gupta #define GPIO_YZAAAB_DATA_VALUE (0x1E0 >> 2)
1624b7f9568SRashmica Gupta #define GPIO_YZAAAB_DIRECTION (0x1E4 >> 2)
1634b7f9568SRashmica Gupta #define GPIO_AC_DATA_VALUE (0x1E8 >> 2)
1644b7f9568SRashmica Gupta #define GPIO_AC_DIRECTION (0x1EC >> 2)
16598edb134SJoel Stanley #define GPIO_3_3V_MEM_SIZE 0x1F0
16698edb134SJoel Stanley #define GPIO_3_3V_REG_ARRAY_SIZE (GPIO_3_3V_MEM_SIZE >> 2)
1674b7f9568SRashmica Gupta
16836d737eeSRashmica Gupta /* AST2600 only - 1.8V gpios */
16936d737eeSRashmica Gupta /*
17098edb134SJoel Stanley * The AST2600 two copies of the GPIO controller: the same 3.3V gpios as the
17164e5758bSJoel Stanley * AST2400 (memory offsets 0x0-0x198) and a second controller with 1.8V gpios
17264e5758bSJoel Stanley * (memory offsets 0x800-0x9D4).
17336d737eeSRashmica Gupta */
17464e5758bSJoel Stanley #define GPIO_1_8V_ABCD_DATA_VALUE (0x000 >> 2)
17564e5758bSJoel Stanley #define GPIO_1_8V_ABCD_DIRECTION (0x004 >> 2)
17664e5758bSJoel Stanley #define GPIO_1_8V_ABCD_INT_ENABLE (0x008 >> 2)
17764e5758bSJoel Stanley #define GPIO_1_8V_ABCD_INT_SENS_0 (0x00C >> 2)
17864e5758bSJoel Stanley #define GPIO_1_8V_ABCD_INT_SENS_1 (0x010 >> 2)
17964e5758bSJoel Stanley #define GPIO_1_8V_ABCD_INT_SENS_2 (0x014 >> 2)
18064e5758bSJoel Stanley #define GPIO_1_8V_ABCD_INT_STATUS (0x018 >> 2)
18164e5758bSJoel Stanley #define GPIO_1_8V_ABCD_RESET_TOLERANT (0x01C >> 2)
18264e5758bSJoel Stanley #define GPIO_1_8V_E_DATA_VALUE (0x020 >> 2)
18364e5758bSJoel Stanley #define GPIO_1_8V_E_DIRECTION (0x024 >> 2)
18464e5758bSJoel Stanley #define GPIO_1_8V_E_INT_ENABLE (0x028 >> 2)
18564e5758bSJoel Stanley #define GPIO_1_8V_E_INT_SENS_0 (0x02C >> 2)
18664e5758bSJoel Stanley #define GPIO_1_8V_E_INT_SENS_1 (0x030 >> 2)
18764e5758bSJoel Stanley #define GPIO_1_8V_E_INT_SENS_2 (0x034 >> 2)
18864e5758bSJoel Stanley #define GPIO_1_8V_E_INT_STATUS (0x038 >> 2)
18964e5758bSJoel Stanley #define GPIO_1_8V_E_RESET_TOLERANT (0x03C >> 2)
19064e5758bSJoel Stanley #define GPIO_1_8V_ABCD_DEBOUNCE_1 (0x040 >> 2)
19164e5758bSJoel Stanley #define GPIO_1_8V_ABCD_DEBOUNCE_2 (0x044 >> 2)
19264e5758bSJoel Stanley #define GPIO_1_8V_E_DEBOUNCE_1 (0x048 >> 2)
19364e5758bSJoel Stanley #define GPIO_1_8V_E_DEBOUNCE_2 (0x04C >> 2)
19464e5758bSJoel Stanley #define GPIO_1_8V_DEBOUNCE_TIME_1 (0x050 >> 2)
19564e5758bSJoel Stanley #define GPIO_1_8V_DEBOUNCE_TIME_2 (0x054 >> 2)
19664e5758bSJoel Stanley #define GPIO_1_8V_DEBOUNCE_TIME_3 (0x058 >> 2)
19764e5758bSJoel Stanley #define GPIO_1_8V_ABCD_COMMAND_SRC_0 (0x060 >> 2)
19864e5758bSJoel Stanley #define GPIO_1_8V_ABCD_COMMAND_SRC_1 (0x064 >> 2)
19964e5758bSJoel Stanley #define GPIO_1_8V_E_COMMAND_SRC_0 (0x068 >> 2)
20064e5758bSJoel Stanley #define GPIO_1_8V_E_COMMAND_SRC_1 (0x06C >> 2)
20164e5758bSJoel Stanley #define GPIO_1_8V_ABCD_DATA_READ (0x0C0 >> 2)
20264e5758bSJoel Stanley #define GPIO_1_8V_E_DATA_READ (0x0C4 >> 2)
20364e5758bSJoel Stanley #define GPIO_1_8V_ABCD_INPUT_MASK (0x1D0 >> 2)
20464e5758bSJoel Stanley #define GPIO_1_8V_E_INPUT_MASK (0x1D4 >> 2)
20564e5758bSJoel Stanley #define GPIO_1_8V_MEM_SIZE 0x1D8
20664e5758bSJoel Stanley #define GPIO_1_8V_REG_ARRAY_SIZE (GPIO_1_8V_MEM_SIZE >> 2)
20736d737eeSRashmica Gupta
208247c0029SJamin Lin /*
209247c0029SJamin Lin * GPIO index mode support
210247c0029SJamin Lin * It only supports write operation
211247c0029SJamin Lin */
212247c0029SJamin Lin REG32(GPIO_INDEX_REG, 0x2AC)
213247c0029SJamin Lin FIELD(GPIO_INDEX_REG, NUMBER, 0, 8)
214247c0029SJamin Lin FIELD(GPIO_INDEX_REG, COMMAND, 12, 1)
215247c0029SJamin Lin FIELD(GPIO_INDEX_REG, TYPE, 16, 4)
216247c0029SJamin Lin FIELD(GPIO_INDEX_REG, DATA_VALUE, 20, 1)
217247c0029SJamin Lin FIELD(GPIO_INDEX_REG, DIRECTION, 20, 1)
218247c0029SJamin Lin FIELD(GPIO_INDEX_REG, INT_ENABLE, 20, 1)
219247c0029SJamin Lin FIELD(GPIO_INDEX_REG, INT_SENS_0, 21, 1)
220247c0029SJamin Lin FIELD(GPIO_INDEX_REG, INT_SENS_1, 22, 1)
221247c0029SJamin Lin FIELD(GPIO_INDEX_REG, INT_SENS_2, 23, 1)
222247c0029SJamin Lin FIELD(GPIO_INDEX_REG, INT_STATUS, 24, 1)
223247c0029SJamin Lin FIELD(GPIO_INDEX_REG, DEBOUNCE_1, 20, 1)
224247c0029SJamin Lin FIELD(GPIO_INDEX_REG, DEBOUNCE_2, 21, 1)
225247c0029SJamin Lin FIELD(GPIO_INDEX_REG, RESET_TOLERANT, 20, 1)
226247c0029SJamin Lin FIELD(GPIO_INDEX_REG, COMMAND_SRC_0, 20, 1)
227247c0029SJamin Lin FIELD(GPIO_INDEX_REG, COMMAND_SRC_1, 21, 1)
228247c0029SJamin Lin FIELD(GPIO_INDEX_REG, INPUT_MASK, 20, 1)
229247c0029SJamin Lin
230bac69883SJamin Lin /* AST2700 GPIO Register Address Offsets */
231bac69883SJamin Lin REG32(GPIO_2700_DEBOUNCE_TIME_1, 0x000)
232bac69883SJamin Lin REG32(GPIO_2700_DEBOUNCE_TIME_2, 0x004)
233bac69883SJamin Lin REG32(GPIO_2700_DEBOUNCE_TIME_3, 0x008)
234bac69883SJamin Lin REG32(GPIO_2700_INT_STATUS_1, 0x100)
235bac69883SJamin Lin REG32(GPIO_2700_INT_STATUS_2, 0x104)
236bac69883SJamin Lin REG32(GPIO_2700_INT_STATUS_3, 0x108)
237bac69883SJamin Lin REG32(GPIO_2700_INT_STATUS_4, 0x10C)
238bac69883SJamin Lin REG32(GPIO_2700_INT_STATUS_5, 0x110)
239bac69883SJamin Lin REG32(GPIO_2700_INT_STATUS_6, 0x114)
240bac69883SJamin Lin REG32(GPIO_2700_INT_STATUS_7, 0x118)
241bac69883SJamin Lin /* GPIOA0 - GPIOAA7 Control Register */
242bac69883SJamin Lin REG32(GPIO_A0_CONTROL, 0x180)
243bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_OUT_DATA, 0, 1)
244bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_DIRECTION, 1, 1)
245bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_INT_ENABLE, 2, 1)
246bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_INT_SENS_0, 3, 1)
247bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_INT_SENS_1, 4, 1)
248bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_INT_SENS_2, 5, 1)
249bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_RESET_TOLERANCE, 6, 1)
250bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_DEBOUNCE_1, 7, 1)
251bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_DEBOUNCE_2, 8, 1)
252bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_INPUT_MASK, 9, 1)
253bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_BLINK_COUNTER_1, 10, 1)
254bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_BLINK_COUNTER_2, 11, 1)
255bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_INT_STATUS, 12, 1)
256bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_IN_DATA, 13, 1)
257bac69883SJamin Lin SHARED_FIELD(GPIO_CONTROL_RESERVED, 14, 18)
258bac69883SJamin Lin REG32(GPIO_AA7_CONTROL, 0x4DC)
259bac69883SJamin Lin #define GPIO_2700_MEM_SIZE 0x4E0
260bac69883SJamin Lin #define GPIO_2700_REG_ARRAY_SIZE (GPIO_2700_MEM_SIZE >> 2)
261bac69883SJamin Lin
aspeed_evaluate_irq(GPIOSets * regs,int gpio_prev_high,int gpio)2624b7f9568SRashmica Gupta static int aspeed_evaluate_irq(GPIOSets *regs, int gpio_prev_high, int gpio)
2634b7f9568SRashmica Gupta {
2644b7f9568SRashmica Gupta uint32_t falling_edge = 0, rising_edge = 0;
2654b7f9568SRashmica Gupta uint32_t int_trigger = extract32(regs->int_sens_0, gpio, 1)
2664b7f9568SRashmica Gupta | extract32(regs->int_sens_1, gpio, 1) << 1
2674b7f9568SRashmica Gupta | extract32(regs->int_sens_2, gpio, 1) << 2;
2684b7f9568SRashmica Gupta uint32_t gpio_curr_high = extract32(regs->data_value, gpio, 1);
2694b7f9568SRashmica Gupta uint32_t gpio_int_enabled = extract32(regs->int_enable, gpio, 1);
2704b7f9568SRashmica Gupta
2714b7f9568SRashmica Gupta if (!gpio_int_enabled) {
2724b7f9568SRashmica Gupta return 0;
2734b7f9568SRashmica Gupta }
2744b7f9568SRashmica Gupta
2754b7f9568SRashmica Gupta /* Detect edges */
2764b7f9568SRashmica Gupta if (gpio_curr_high && !gpio_prev_high) {
2774b7f9568SRashmica Gupta rising_edge = 1;
2784b7f9568SRashmica Gupta } else if (!gpio_curr_high && gpio_prev_high) {
2794b7f9568SRashmica Gupta falling_edge = 1;
2804b7f9568SRashmica Gupta }
2814b7f9568SRashmica Gupta
2824b7f9568SRashmica Gupta if (((int_trigger == ASPEED_FALLING_EDGE) && falling_edge) ||
2834b7f9568SRashmica Gupta ((int_trigger == ASPEED_RISING_EDGE) && rising_edge) ||
2844b7f9568SRashmica Gupta ((int_trigger == ASPEED_LEVEL_LOW) && !gpio_curr_high) ||
2854b7f9568SRashmica Gupta ((int_trigger == ASPEED_LEVEL_HIGH) && gpio_curr_high) ||
2864b7f9568SRashmica Gupta ((int_trigger >= ASPEED_DUAL_EDGE) && (rising_edge || falling_edge)))
2874b7f9568SRashmica Gupta {
2884b7f9568SRashmica Gupta regs->int_status = deposit32(regs->int_status, gpio, 1, 1);
2894b7f9568SRashmica Gupta return 1;
2904b7f9568SRashmica Gupta }
2914b7f9568SRashmica Gupta return 0;
2924b7f9568SRashmica Gupta }
2934b7f9568SRashmica Gupta
2944b7f9568SRashmica Gupta #define nested_struct_index(ta, pa, m, tb, pb) \
2954b7f9568SRashmica Gupta (pb - ((tb *)(((char *)pa) + offsetof(ta, m))))
2964b7f9568SRashmica Gupta
aspeed_gpio_set_idx(AspeedGPIOState * s,GPIOSets * regs)2974b7f9568SRashmica Gupta static ptrdiff_t aspeed_gpio_set_idx(AspeedGPIOState *s, GPIOSets *regs)
2984b7f9568SRashmica Gupta {
2994b7f9568SRashmica Gupta return nested_struct_index(AspeedGPIOState, s, sets, GPIOSets, regs);
3004b7f9568SRashmica Gupta }
3014b7f9568SRashmica Gupta
aspeed_gpio_update(AspeedGPIOState * s,GPIOSets * regs,uint32_t value,uint32_t mode_mask)3024b7f9568SRashmica Gupta static void aspeed_gpio_update(AspeedGPIOState *s, GPIOSets *regs,
3031f30db92SPeter Delevoryas uint32_t value, uint32_t mode_mask)
3044b7f9568SRashmica Gupta {
3054b7f9568SRashmica Gupta uint32_t input_mask = regs->input_mask;
3064b7f9568SRashmica Gupta uint32_t direction = regs->direction;
3074b7f9568SRashmica Gupta uint32_t old = regs->data_value;
3084b7f9568SRashmica Gupta uint32_t new = value;
3094b7f9568SRashmica Gupta uint32_t diff;
3104b7f9568SRashmica Gupta int gpio;
3114b7f9568SRashmica Gupta
3121f30db92SPeter Delevoryas diff = (old ^ new);
3131f30db92SPeter Delevoryas diff &= mode_mask;
3144b7f9568SRashmica Gupta if (diff) {
31587bd33e8SPeter Delevoryas for (gpio = 0; gpio < ASPEED_GPIOS_PER_SET; gpio++) {
316737cb2f3SPeter Maydell uint32_t mask = 1U << gpio;
3174b7f9568SRashmica Gupta
3184b7f9568SRashmica Gupta /* If the gpio needs to be updated... */
3194b7f9568SRashmica Gupta if (!(diff & mask)) {
3204b7f9568SRashmica Gupta continue;
3214b7f9568SRashmica Gupta }
3224b7f9568SRashmica Gupta
3234b7f9568SRashmica Gupta /* ...and we're output or not input-masked... */
3244b7f9568SRashmica Gupta if (!(direction & mask) && (input_mask & mask)) {
3254b7f9568SRashmica Gupta continue;
3264b7f9568SRashmica Gupta }
3274b7f9568SRashmica Gupta
3284b7f9568SRashmica Gupta /* ...then update the state. */
3294b7f9568SRashmica Gupta if (mask & new) {
3304b7f9568SRashmica Gupta regs->data_value |= mask;
3314b7f9568SRashmica Gupta } else {
3324b7f9568SRashmica Gupta regs->data_value &= ~mask;
3334b7f9568SRashmica Gupta }
3344b7f9568SRashmica Gupta
3354b7f9568SRashmica Gupta /* If the gpio is set to output... */
3364b7f9568SRashmica Gupta if (direction & mask) {
3374b7f9568SRashmica Gupta /* ...trigger the line-state IRQ */
3384b7f9568SRashmica Gupta ptrdiff_t set = aspeed_gpio_set_idx(s, regs);
33987bd33e8SPeter Delevoryas qemu_set_irq(s->gpios[set][gpio], !!(new & mask));
3404b7f9568SRashmica Gupta } else {
3414b7f9568SRashmica Gupta /* ...otherwise if we meet the line's current IRQ policy... */
3424b7f9568SRashmica Gupta if (aspeed_evaluate_irq(regs, old & mask, gpio)) {
3434b7f9568SRashmica Gupta /* ...trigger the VIC IRQ */
3444b7f9568SRashmica Gupta s->pending++;
3454b7f9568SRashmica Gupta }
3464b7f9568SRashmica Gupta }
3474b7f9568SRashmica Gupta }
3484b7f9568SRashmica Gupta }
3494b7f9568SRashmica Gupta qemu_set_irq(s->irq, !!(s->pending));
3504b7f9568SRashmica Gupta }
3514b7f9568SRashmica Gupta
aspeed_gpio_get_pin_level(AspeedGPIOState * s,uint32_t set_idx,uint32_t pin)3524b7f9568SRashmica Gupta static bool aspeed_gpio_get_pin_level(AspeedGPIOState *s, uint32_t set_idx,
3534b7f9568SRashmica Gupta uint32_t pin)
3544b7f9568SRashmica Gupta {
3554b7f9568SRashmica Gupta uint32_t reg_val;
3564b7f9568SRashmica Gupta uint32_t pin_mask = 1 << pin;
3574b7f9568SRashmica Gupta
3584b7f9568SRashmica Gupta reg_val = s->sets[set_idx].data_value;
3594b7f9568SRashmica Gupta
3604b7f9568SRashmica Gupta return !!(reg_val & pin_mask);
3614b7f9568SRashmica Gupta }
3624b7f9568SRashmica Gupta
aspeed_gpio_set_pin_level(AspeedGPIOState * s,uint32_t set_idx,uint32_t pin,bool level)3634b7f9568SRashmica Gupta static void aspeed_gpio_set_pin_level(AspeedGPIOState *s, uint32_t set_idx,
3644b7f9568SRashmica Gupta uint32_t pin, bool level)
3654b7f9568SRashmica Gupta {
3664b7f9568SRashmica Gupta uint32_t value = s->sets[set_idx].data_value;
3674b7f9568SRashmica Gupta uint32_t pin_mask = 1 << pin;
3684b7f9568SRashmica Gupta
3694b7f9568SRashmica Gupta if (level) {
3704b7f9568SRashmica Gupta value |= pin_mask;
3714b7f9568SRashmica Gupta } else {
3722ec06378SPeter Delevoryas value &= ~pin_mask;
3734b7f9568SRashmica Gupta }
3744b7f9568SRashmica Gupta
37533343bffSJamin Lin aspeed_gpio_update(s, &s->sets[set_idx], value,
37633343bffSJamin Lin ~s->sets[set_idx].direction);
3774b7f9568SRashmica Gupta }
3784b7f9568SRashmica Gupta
3794b7f9568SRashmica Gupta /*
3804b7f9568SRashmica Gupta * | src_1 | src_2 | source |
3814b7f9568SRashmica Gupta * |-----------------------------|
3824b7f9568SRashmica Gupta * | 0 | 0 | ARM |
3834b7f9568SRashmica Gupta * | 0 | 1 | LPC |
3844b7f9568SRashmica Gupta * | 1 | 0 | Coprocessor|
3854b7f9568SRashmica Gupta * | 1 | 1 | Reserved |
3864b7f9568SRashmica Gupta *
3874b7f9568SRashmica Gupta * Once the source of a set is programmed, corresponding bits in the
3884b7f9568SRashmica Gupta * data_value, direction, interrupt [enable, sens[0-2]], reset_tol and
3894b7f9568SRashmica Gupta * debounce registers can only be written by the source.
3904b7f9568SRashmica Gupta *
3914b7f9568SRashmica Gupta * Source is ARM by default
3924b7f9568SRashmica Gupta * only bits 24, 16, 8, and 0 can be set
3934b7f9568SRashmica Gupta *
3944b7f9568SRashmica Gupta * we don't currently have a model for the LPC or Coprocessor
3954b7f9568SRashmica Gupta */
update_value_control_source(GPIOSets * regs,uint32_t old_value,uint32_t value)3964b7f9568SRashmica Gupta static uint32_t update_value_control_source(GPIOSets *regs, uint32_t old_value,
3974b7f9568SRashmica Gupta uint32_t value)
3984b7f9568SRashmica Gupta {
3994b7f9568SRashmica Gupta int i;
4004b7f9568SRashmica Gupta int cmd_source;
4014b7f9568SRashmica Gupta
4024b7f9568SRashmica Gupta /* assume the source is always ARM for now */
4034b7f9568SRashmica Gupta int source = ASPEED_SOURCE_ARM;
4044b7f9568SRashmica Gupta
4054b7f9568SRashmica Gupta uint32_t new_value = 0;
4064b7f9568SRashmica Gupta
4074b7f9568SRashmica Gupta /* for each group in set */
40887bd33e8SPeter Delevoryas for (i = 0; i < ASPEED_GPIOS_PER_SET; i += GPIOS_PER_GROUP) {
4094b7f9568SRashmica Gupta cmd_source = extract32(regs->cmd_source_0, i, 1)
4104b7f9568SRashmica Gupta | (extract32(regs->cmd_source_1, i, 1) << 1);
4114b7f9568SRashmica Gupta
4124b7f9568SRashmica Gupta if (source == cmd_source) {
4134b7f9568SRashmica Gupta new_value |= (0xff << i) & value;
4144b7f9568SRashmica Gupta } else {
4154b7f9568SRashmica Gupta new_value |= (0xff << i) & old_value;
4164b7f9568SRashmica Gupta }
4174b7f9568SRashmica Gupta }
4184b7f9568SRashmica Gupta return new_value;
4194b7f9568SRashmica Gupta }
4204b7f9568SRashmica Gupta
42198edb134SJoel Stanley static const AspeedGPIOReg aspeed_3_3v_gpios[GPIO_3_3V_REG_ARRAY_SIZE] = {
4224b7f9568SRashmica Gupta /* Set ABCD */
4234b7f9568SRashmica Gupta [GPIO_ABCD_DATA_VALUE] = { 0, gpio_reg_data_value },
4244b7f9568SRashmica Gupta [GPIO_ABCD_DIRECTION] = { 0, gpio_reg_direction },
4254b7f9568SRashmica Gupta [GPIO_ABCD_INT_ENABLE] = { 0, gpio_reg_int_enable },
4264b7f9568SRashmica Gupta [GPIO_ABCD_INT_SENS_0] = { 0, gpio_reg_int_sens_0 },
4274b7f9568SRashmica Gupta [GPIO_ABCD_INT_SENS_1] = { 0, gpio_reg_int_sens_1 },
4284b7f9568SRashmica Gupta [GPIO_ABCD_INT_SENS_2] = { 0, gpio_reg_int_sens_2 },
4294b7f9568SRashmica Gupta [GPIO_ABCD_INT_STATUS] = { 0, gpio_reg_int_status },
4304b7f9568SRashmica Gupta [GPIO_ABCD_RESET_TOLERANT] = { 0, gpio_reg_reset_tolerant },
4314b7f9568SRashmica Gupta [GPIO_ABCD_DEBOUNCE_1] = { 0, gpio_reg_debounce_1 },
4324b7f9568SRashmica Gupta [GPIO_ABCD_DEBOUNCE_2] = { 0, gpio_reg_debounce_2 },
4334b7f9568SRashmica Gupta [GPIO_ABCD_COMMAND_SRC_0] = { 0, gpio_reg_cmd_source_0 },
4344b7f9568SRashmica Gupta [GPIO_ABCD_COMMAND_SRC_1] = { 0, gpio_reg_cmd_source_1 },
4354b7f9568SRashmica Gupta [GPIO_ABCD_DATA_READ] = { 0, gpio_reg_data_read },
4364b7f9568SRashmica Gupta [GPIO_ABCD_INPUT_MASK] = { 0, gpio_reg_input_mask },
4374b7f9568SRashmica Gupta /* Set EFGH */
4384b7f9568SRashmica Gupta [GPIO_EFGH_DATA_VALUE] = { 1, gpio_reg_data_value },
4394b7f9568SRashmica Gupta [GPIO_EFGH_DIRECTION] = { 1, gpio_reg_direction },
4404b7f9568SRashmica Gupta [GPIO_EFGH_INT_ENABLE] = { 1, gpio_reg_int_enable },
4414b7f9568SRashmica Gupta [GPIO_EFGH_INT_SENS_0] = { 1, gpio_reg_int_sens_0 },
4424b7f9568SRashmica Gupta [GPIO_EFGH_INT_SENS_1] = { 1, gpio_reg_int_sens_1 },
4434b7f9568SRashmica Gupta [GPIO_EFGH_INT_SENS_2] = { 1, gpio_reg_int_sens_2 },
4444b7f9568SRashmica Gupta [GPIO_EFGH_INT_STATUS] = { 1, gpio_reg_int_status },
4454b7f9568SRashmica Gupta [GPIO_EFGH_RESET_TOLERANT] = { 1, gpio_reg_reset_tolerant },
4464b7f9568SRashmica Gupta [GPIO_EFGH_DEBOUNCE_1] = { 1, gpio_reg_debounce_1 },
4474b7f9568SRashmica Gupta [GPIO_EFGH_DEBOUNCE_2] = { 1, gpio_reg_debounce_2 },
4484b7f9568SRashmica Gupta [GPIO_EFGH_COMMAND_SRC_0] = { 1, gpio_reg_cmd_source_0 },
4494b7f9568SRashmica Gupta [GPIO_EFGH_COMMAND_SRC_1] = { 1, gpio_reg_cmd_source_1 },
4504b7f9568SRashmica Gupta [GPIO_EFGH_DATA_READ] = { 1, gpio_reg_data_read },
4514b7f9568SRashmica Gupta [GPIO_EFGH_INPUT_MASK] = { 1, gpio_reg_input_mask },
4524b7f9568SRashmica Gupta /* Set IJKL */
4534b7f9568SRashmica Gupta [GPIO_IJKL_DATA_VALUE] = { 2, gpio_reg_data_value },
4544b7f9568SRashmica Gupta [GPIO_IJKL_DIRECTION] = { 2, gpio_reg_direction },
4554b7f9568SRashmica Gupta [GPIO_IJKL_INT_ENABLE] = { 2, gpio_reg_int_enable },
4564b7f9568SRashmica Gupta [GPIO_IJKL_INT_SENS_0] = { 2, gpio_reg_int_sens_0 },
4574b7f9568SRashmica Gupta [GPIO_IJKL_INT_SENS_1] = { 2, gpio_reg_int_sens_1 },
4584b7f9568SRashmica Gupta [GPIO_IJKL_INT_SENS_2] = { 2, gpio_reg_int_sens_2 },
4594b7f9568SRashmica Gupta [GPIO_IJKL_INT_STATUS] = { 2, gpio_reg_int_status },
4604b7f9568SRashmica Gupta [GPIO_IJKL_RESET_TOLERANT] = { 2, gpio_reg_reset_tolerant },
4614b7f9568SRashmica Gupta [GPIO_IJKL_DEBOUNCE_1] = { 2, gpio_reg_debounce_1 },
4624b7f9568SRashmica Gupta [GPIO_IJKL_DEBOUNCE_2] = { 2, gpio_reg_debounce_2 },
4634b7f9568SRashmica Gupta [GPIO_IJKL_COMMAND_SRC_0] = { 2, gpio_reg_cmd_source_0 },
4644b7f9568SRashmica Gupta [GPIO_IJKL_COMMAND_SRC_1] = { 2, gpio_reg_cmd_source_1 },
4654b7f9568SRashmica Gupta [GPIO_IJKL_DATA_READ] = { 2, gpio_reg_data_read },
4664b7f9568SRashmica Gupta [GPIO_IJKL_INPUT_MASK] = { 2, gpio_reg_input_mask },
4674b7f9568SRashmica Gupta /* Set MNOP */
4684b7f9568SRashmica Gupta [GPIO_MNOP_DATA_VALUE] = { 3, gpio_reg_data_value },
4694b7f9568SRashmica Gupta [GPIO_MNOP_DIRECTION] = { 3, gpio_reg_direction },
4704b7f9568SRashmica Gupta [GPIO_MNOP_INT_ENABLE] = { 3, gpio_reg_int_enable },
4714b7f9568SRashmica Gupta [GPIO_MNOP_INT_SENS_0] = { 3, gpio_reg_int_sens_0 },
4724b7f9568SRashmica Gupta [GPIO_MNOP_INT_SENS_1] = { 3, gpio_reg_int_sens_1 },
4734b7f9568SRashmica Gupta [GPIO_MNOP_INT_SENS_2] = { 3, gpio_reg_int_sens_2 },
4744b7f9568SRashmica Gupta [GPIO_MNOP_INT_STATUS] = { 3, gpio_reg_int_status },
4754b7f9568SRashmica Gupta [GPIO_MNOP_RESET_TOLERANT] = { 3, gpio_reg_reset_tolerant },
4764b7f9568SRashmica Gupta [GPIO_MNOP_DEBOUNCE_1] = { 3, gpio_reg_debounce_1 },
4774b7f9568SRashmica Gupta [GPIO_MNOP_DEBOUNCE_2] = { 3, gpio_reg_debounce_2 },
4784b7f9568SRashmica Gupta [GPIO_MNOP_COMMAND_SRC_0] = { 3, gpio_reg_cmd_source_0 },
4794b7f9568SRashmica Gupta [GPIO_MNOP_COMMAND_SRC_1] = { 3, gpio_reg_cmd_source_1 },
4804b7f9568SRashmica Gupta [GPIO_MNOP_DATA_READ] = { 3, gpio_reg_data_read },
4814b7f9568SRashmica Gupta [GPIO_MNOP_INPUT_MASK] = { 3, gpio_reg_input_mask },
4824b7f9568SRashmica Gupta /* Set QRST */
4834b7f9568SRashmica Gupta [GPIO_QRST_DATA_VALUE] = { 4, gpio_reg_data_value },
4844b7f9568SRashmica Gupta [GPIO_QRST_DIRECTION] = { 4, gpio_reg_direction },
4854b7f9568SRashmica Gupta [GPIO_QRST_INT_ENABLE] = { 4, gpio_reg_int_enable },
4864b7f9568SRashmica Gupta [GPIO_QRST_INT_SENS_0] = { 4, gpio_reg_int_sens_0 },
4874b7f9568SRashmica Gupta [GPIO_QRST_INT_SENS_1] = { 4, gpio_reg_int_sens_1 },
4884b7f9568SRashmica Gupta [GPIO_QRST_INT_SENS_2] = { 4, gpio_reg_int_sens_2 },
4894b7f9568SRashmica Gupta [GPIO_QRST_INT_STATUS] = { 4, gpio_reg_int_status },
4904b7f9568SRashmica Gupta [GPIO_QRST_RESET_TOLERANT] = { 4, gpio_reg_reset_tolerant },
4914b7f9568SRashmica Gupta [GPIO_QRST_DEBOUNCE_1] = { 4, gpio_reg_debounce_1 },
4924b7f9568SRashmica Gupta [GPIO_QRST_DEBOUNCE_2] = { 4, gpio_reg_debounce_2 },
4934b7f9568SRashmica Gupta [GPIO_QRST_COMMAND_SRC_0] = { 4, gpio_reg_cmd_source_0 },
4944b7f9568SRashmica Gupta [GPIO_QRST_COMMAND_SRC_1] = { 4, gpio_reg_cmd_source_1 },
4954b7f9568SRashmica Gupta [GPIO_QRST_DATA_READ] = { 4, gpio_reg_data_read },
4964b7f9568SRashmica Gupta [GPIO_QRST_INPUT_MASK] = { 4, gpio_reg_input_mask },
4974b7f9568SRashmica Gupta /* Set UVWX */
4984b7f9568SRashmica Gupta [GPIO_UVWX_DATA_VALUE] = { 5, gpio_reg_data_value },
4994b7f9568SRashmica Gupta [GPIO_UVWX_DIRECTION] = { 5, gpio_reg_direction },
5004b7f9568SRashmica Gupta [GPIO_UVWX_INT_ENABLE] = { 5, gpio_reg_int_enable },
5014b7f9568SRashmica Gupta [GPIO_UVWX_INT_SENS_0] = { 5, gpio_reg_int_sens_0 },
5024b7f9568SRashmica Gupta [GPIO_UVWX_INT_SENS_1] = { 5, gpio_reg_int_sens_1 },
5034b7f9568SRashmica Gupta [GPIO_UVWX_INT_SENS_2] = { 5, gpio_reg_int_sens_2 },
5044b7f9568SRashmica Gupta [GPIO_UVWX_INT_STATUS] = { 5, gpio_reg_int_status },
5054b7f9568SRashmica Gupta [GPIO_UVWX_RESET_TOLERANT] = { 5, gpio_reg_reset_tolerant },
5064b7f9568SRashmica Gupta [GPIO_UVWX_DEBOUNCE_1] = { 5, gpio_reg_debounce_1 },
5074b7f9568SRashmica Gupta [GPIO_UVWX_DEBOUNCE_2] = { 5, gpio_reg_debounce_2 },
5084b7f9568SRashmica Gupta [GPIO_UVWX_COMMAND_SRC_0] = { 5, gpio_reg_cmd_source_0 },
5094b7f9568SRashmica Gupta [GPIO_UVWX_COMMAND_SRC_1] = { 5, gpio_reg_cmd_source_1 },
5104b7f9568SRashmica Gupta [GPIO_UVWX_DATA_READ] = { 5, gpio_reg_data_read },
5114b7f9568SRashmica Gupta [GPIO_UVWX_INPUT_MASK] = { 5, gpio_reg_input_mask },
5124b7f9568SRashmica Gupta /* Set YZAAAB */
5134b7f9568SRashmica Gupta [GPIO_YZAAAB_DATA_VALUE] = { 6, gpio_reg_data_value },
5144b7f9568SRashmica Gupta [GPIO_YZAAAB_DIRECTION] = { 6, gpio_reg_direction },
5154b7f9568SRashmica Gupta [GPIO_YZAAAB_INT_ENABLE] = { 6, gpio_reg_int_enable },
5164b7f9568SRashmica Gupta [GPIO_YZAAAB_INT_SENS_0] = { 6, gpio_reg_int_sens_0 },
5174b7f9568SRashmica Gupta [GPIO_YZAAAB_INT_SENS_1] = { 6, gpio_reg_int_sens_1 },
5184b7f9568SRashmica Gupta [GPIO_YZAAAB_INT_SENS_2] = { 6, gpio_reg_int_sens_2 },
5194b7f9568SRashmica Gupta [GPIO_YZAAAB_INT_STATUS] = { 6, gpio_reg_int_status },
5204b7f9568SRashmica Gupta [GPIO_YZAAAB_RESET_TOLERANT] = { 6, gpio_reg_reset_tolerant },
5214b7f9568SRashmica Gupta [GPIO_YZAAAB_DEBOUNCE_1] = { 6, gpio_reg_debounce_1 },
5224b7f9568SRashmica Gupta [GPIO_YZAAAB_DEBOUNCE_2] = { 6, gpio_reg_debounce_2 },
5234b7f9568SRashmica Gupta [GPIO_YZAAAB_COMMAND_SRC_0] = { 6, gpio_reg_cmd_source_0 },
5244b7f9568SRashmica Gupta [GPIO_YZAAAB_COMMAND_SRC_1] = { 6, gpio_reg_cmd_source_1 },
5254b7f9568SRashmica Gupta [GPIO_YZAAAB_DATA_READ] = { 6, gpio_reg_data_read },
5264b7f9568SRashmica Gupta [GPIO_YZAAAB_INPUT_MASK] = { 6, gpio_reg_input_mask },
5274b7f9568SRashmica Gupta /* Set AC (ast2500 only) */
5284b7f9568SRashmica Gupta [GPIO_AC_DATA_VALUE] = { 7, gpio_reg_data_value },
5294b7f9568SRashmica Gupta [GPIO_AC_DIRECTION] = { 7, gpio_reg_direction },
5304b7f9568SRashmica Gupta [GPIO_AC_INT_ENABLE] = { 7, gpio_reg_int_enable },
5314b7f9568SRashmica Gupta [GPIO_AC_INT_SENS_0] = { 7, gpio_reg_int_sens_0 },
5324b7f9568SRashmica Gupta [GPIO_AC_INT_SENS_1] = { 7, gpio_reg_int_sens_1 },
5334b7f9568SRashmica Gupta [GPIO_AC_INT_SENS_2] = { 7, gpio_reg_int_sens_2 },
5344b7f9568SRashmica Gupta [GPIO_AC_INT_STATUS] = { 7, gpio_reg_int_status },
5354b7f9568SRashmica Gupta [GPIO_AC_RESET_TOLERANT] = { 7, gpio_reg_reset_tolerant },
5364b7f9568SRashmica Gupta [GPIO_AC_DEBOUNCE_1] = { 7, gpio_reg_debounce_1 },
5374b7f9568SRashmica Gupta [GPIO_AC_DEBOUNCE_2] = { 7, gpio_reg_debounce_2 },
5384b7f9568SRashmica Gupta [GPIO_AC_COMMAND_SRC_0] = { 7, gpio_reg_cmd_source_0 },
5394b7f9568SRashmica Gupta [GPIO_AC_COMMAND_SRC_1] = { 7, gpio_reg_cmd_source_1 },
5404b7f9568SRashmica Gupta [GPIO_AC_DATA_READ] = { 7, gpio_reg_data_read },
5414b7f9568SRashmica Gupta [GPIO_AC_INPUT_MASK] = { 7, gpio_reg_input_mask },
5424b7f9568SRashmica Gupta };
5434b7f9568SRashmica Gupta
54436d737eeSRashmica Gupta static const AspeedGPIOReg aspeed_1_8v_gpios[GPIO_1_8V_REG_ARRAY_SIZE] = {
54536d737eeSRashmica Gupta /* 1.8V Set ABCD */
54636d737eeSRashmica Gupta [GPIO_1_8V_ABCD_DATA_VALUE] = {0, gpio_reg_data_value},
54736d737eeSRashmica Gupta [GPIO_1_8V_ABCD_DIRECTION] = {0, gpio_reg_direction},
54836d737eeSRashmica Gupta [GPIO_1_8V_ABCD_INT_ENABLE] = {0, gpio_reg_int_enable},
54936d737eeSRashmica Gupta [GPIO_1_8V_ABCD_INT_SENS_0] = {0, gpio_reg_int_sens_0},
55036d737eeSRashmica Gupta [GPIO_1_8V_ABCD_INT_SENS_1] = {0, gpio_reg_int_sens_1},
55136d737eeSRashmica Gupta [GPIO_1_8V_ABCD_INT_SENS_2] = {0, gpio_reg_int_sens_2},
55236d737eeSRashmica Gupta [GPIO_1_8V_ABCD_INT_STATUS] = {0, gpio_reg_int_status},
55336d737eeSRashmica Gupta [GPIO_1_8V_ABCD_RESET_TOLERANT] = {0, gpio_reg_reset_tolerant},
55436d737eeSRashmica Gupta [GPIO_1_8V_ABCD_DEBOUNCE_1] = {0, gpio_reg_debounce_1},
55536d737eeSRashmica Gupta [GPIO_1_8V_ABCD_DEBOUNCE_2] = {0, gpio_reg_debounce_2},
55636d737eeSRashmica Gupta [GPIO_1_8V_ABCD_COMMAND_SRC_0] = {0, gpio_reg_cmd_source_0},
55736d737eeSRashmica Gupta [GPIO_1_8V_ABCD_COMMAND_SRC_1] = {0, gpio_reg_cmd_source_1},
55836d737eeSRashmica Gupta [GPIO_1_8V_ABCD_DATA_READ] = {0, gpio_reg_data_read},
55936d737eeSRashmica Gupta [GPIO_1_8V_ABCD_INPUT_MASK] = {0, gpio_reg_input_mask},
56036d737eeSRashmica Gupta /* 1.8V Set E */
56136d737eeSRashmica Gupta [GPIO_1_8V_E_DATA_VALUE] = {1, gpio_reg_data_value},
56236d737eeSRashmica Gupta [GPIO_1_8V_E_DIRECTION] = {1, gpio_reg_direction},
56336d737eeSRashmica Gupta [GPIO_1_8V_E_INT_ENABLE] = {1, gpio_reg_int_enable},
56436d737eeSRashmica Gupta [GPIO_1_8V_E_INT_SENS_0] = {1, gpio_reg_int_sens_0},
56536d737eeSRashmica Gupta [GPIO_1_8V_E_INT_SENS_1] = {1, gpio_reg_int_sens_1},
56636d737eeSRashmica Gupta [GPIO_1_8V_E_INT_SENS_2] = {1, gpio_reg_int_sens_2},
56736d737eeSRashmica Gupta [GPIO_1_8V_E_INT_STATUS] = {1, gpio_reg_int_status},
56836d737eeSRashmica Gupta [GPIO_1_8V_E_RESET_TOLERANT] = {1, gpio_reg_reset_tolerant},
56936d737eeSRashmica Gupta [GPIO_1_8V_E_DEBOUNCE_1] = {1, gpio_reg_debounce_1},
57036d737eeSRashmica Gupta [GPIO_1_8V_E_DEBOUNCE_2] = {1, gpio_reg_debounce_2},
57136d737eeSRashmica Gupta [GPIO_1_8V_E_COMMAND_SRC_0] = {1, gpio_reg_cmd_source_0},
57236d737eeSRashmica Gupta [GPIO_1_8V_E_COMMAND_SRC_1] = {1, gpio_reg_cmd_source_1},
57336d737eeSRashmica Gupta [GPIO_1_8V_E_DATA_READ] = {1, gpio_reg_data_read},
57436d737eeSRashmica Gupta [GPIO_1_8V_E_INPUT_MASK] = {1, gpio_reg_input_mask},
57536d737eeSRashmica Gupta };
57636d737eeSRashmica Gupta
aspeed_gpio_read(void * opaque,hwaddr offset,uint32_t size)5774b7f9568SRashmica Gupta static uint64_t aspeed_gpio_read(void *opaque, hwaddr offset, uint32_t size)
5784b7f9568SRashmica Gupta {
5794b7f9568SRashmica Gupta AspeedGPIOState *s = ASPEED_GPIO(opaque);
5804b7f9568SRashmica Gupta AspeedGPIOClass *agc = ASPEED_GPIO_GET_CLASS(s);
5814b7f9568SRashmica Gupta uint64_t idx = -1;
5824b7f9568SRashmica Gupta const AspeedGPIOReg *reg;
5834b7f9568SRashmica Gupta GPIOSets *set;
5847b1d21a8SJamin Lin uint32_t value = 0;
5857b1d21a8SJamin Lin uint64_t debounce_value;
5864b7f9568SRashmica Gupta
5874b7f9568SRashmica Gupta idx = offset >> 2;
5884b7f9568SRashmica Gupta if (idx >= GPIO_DEBOUNCE_TIME_1 && idx <= GPIO_DEBOUNCE_TIME_3) {
5894b7f9568SRashmica Gupta idx -= GPIO_DEBOUNCE_TIME_1;
5907b1d21a8SJamin Lin debounce_value = (uint64_t) s->debounce_regs[idx];
5917b1d21a8SJamin Lin trace_aspeed_gpio_read(offset, debounce_value);
5927b1d21a8SJamin Lin return debounce_value;
5934b7f9568SRashmica Gupta }
5944b7f9568SRashmica Gupta
59587511bb8SZheyu Ma if (idx >= agc->reg_table_count) {
59687511bb8SZheyu Ma qemu_log_mask(LOG_GUEST_ERROR, "%s: idx 0x%" PRIx64 " out of bounds\n",
59787511bb8SZheyu Ma __func__, idx);
59887511bb8SZheyu Ma return 0;
59987511bb8SZheyu Ma }
60087511bb8SZheyu Ma
6014b7f9568SRashmica Gupta reg = &agc->reg_table[idx];
6024b7f9568SRashmica Gupta if (reg->set_idx >= agc->nr_gpio_sets) {
6034b7f9568SRashmica Gupta qemu_log_mask(LOG_GUEST_ERROR, "%s: no getter for offset 0x%"
604554c2945SJamin Lin PRIx64"\n", __func__, offset);
6054b7f9568SRashmica Gupta return 0;
6064b7f9568SRashmica Gupta }
6074b7f9568SRashmica Gupta
6084b7f9568SRashmica Gupta set = &s->sets[reg->set_idx];
6094b7f9568SRashmica Gupta switch (reg->type) {
6104b7f9568SRashmica Gupta case gpio_reg_data_value:
6117b1d21a8SJamin Lin value = set->data_value;
6127b1d21a8SJamin Lin break;
6134b7f9568SRashmica Gupta case gpio_reg_direction:
6147b1d21a8SJamin Lin value = set->direction;
6157b1d21a8SJamin Lin break;
6164b7f9568SRashmica Gupta case gpio_reg_int_enable:
6177b1d21a8SJamin Lin value = set->int_enable;
6187b1d21a8SJamin Lin break;
6194b7f9568SRashmica Gupta case gpio_reg_int_sens_0:
6207b1d21a8SJamin Lin value = set->int_sens_0;
6217b1d21a8SJamin Lin break;
6224b7f9568SRashmica Gupta case gpio_reg_int_sens_1:
6237b1d21a8SJamin Lin value = set->int_sens_1;
6247b1d21a8SJamin Lin break;
6254b7f9568SRashmica Gupta case gpio_reg_int_sens_2:
6267b1d21a8SJamin Lin value = set->int_sens_2;
6277b1d21a8SJamin Lin break;
6284b7f9568SRashmica Gupta case gpio_reg_int_status:
6297b1d21a8SJamin Lin value = set->int_status;
6307b1d21a8SJamin Lin break;
6314b7f9568SRashmica Gupta case gpio_reg_reset_tolerant:
6327b1d21a8SJamin Lin value = set->reset_tol;
6337b1d21a8SJamin Lin break;
6344b7f9568SRashmica Gupta case gpio_reg_debounce_1:
6357b1d21a8SJamin Lin value = set->debounce_1;
6367b1d21a8SJamin Lin break;
6374b7f9568SRashmica Gupta case gpio_reg_debounce_2:
6387b1d21a8SJamin Lin value = set->debounce_2;
6397b1d21a8SJamin Lin break;
6404b7f9568SRashmica Gupta case gpio_reg_cmd_source_0:
6417b1d21a8SJamin Lin value = set->cmd_source_0;
6427b1d21a8SJamin Lin break;
6434b7f9568SRashmica Gupta case gpio_reg_cmd_source_1:
6447b1d21a8SJamin Lin value = set->cmd_source_1;
6457b1d21a8SJamin Lin break;
6464b7f9568SRashmica Gupta case gpio_reg_data_read:
6477b1d21a8SJamin Lin value = set->data_read;
6487b1d21a8SJamin Lin break;
6494b7f9568SRashmica Gupta case gpio_reg_input_mask:
6507b1d21a8SJamin Lin value = set->input_mask;
6517b1d21a8SJamin Lin break;
6524b7f9568SRashmica Gupta default:
6534b7f9568SRashmica Gupta qemu_log_mask(LOG_GUEST_ERROR, "%s: no getter for offset 0x%"
654554c2945SJamin Lin PRIx64"\n", __func__, offset);
6554b7f9568SRashmica Gupta return 0;
65646179776SAndrew Jeffery }
6577b1d21a8SJamin Lin
6587b1d21a8SJamin Lin trace_aspeed_gpio_read(offset, value);
6597b1d21a8SJamin Lin return value;
6604b7f9568SRashmica Gupta }
6614b7f9568SRashmica Gupta
aspeed_gpio_write_index_mode(void * opaque,hwaddr offset,uint64_t data,uint32_t size)662247c0029SJamin Lin static void aspeed_gpio_write_index_mode(void *opaque, hwaddr offset,
663247c0029SJamin Lin uint64_t data, uint32_t size)
664247c0029SJamin Lin {
665247c0029SJamin Lin AspeedGPIOState *s = ASPEED_GPIO(opaque);
666247c0029SJamin Lin AspeedGPIOClass *agc = ASPEED_GPIO_GET_CLASS(s);
667247c0029SJamin Lin const GPIOSetProperties *props;
668247c0029SJamin Lin GPIOSets *set;
669247c0029SJamin Lin uint32_t reg_idx_number = FIELD_EX32(data, GPIO_INDEX_REG, NUMBER);
670247c0029SJamin Lin uint32_t reg_idx_type = FIELD_EX32(data, GPIO_INDEX_REG, TYPE);
671247c0029SJamin Lin uint32_t reg_idx_command = FIELD_EX32(data, GPIO_INDEX_REG, COMMAND);
672247c0029SJamin Lin uint32_t set_idx = reg_idx_number / ASPEED_GPIOS_PER_SET;
673247c0029SJamin Lin uint32_t pin_idx = reg_idx_number % ASPEED_GPIOS_PER_SET;
674247c0029SJamin Lin uint32_t group_idx = pin_idx / GPIOS_PER_GROUP;
675247c0029SJamin Lin uint32_t reg_value = 0;
6767e22f6faSJamin Lin uint32_t pending = 0;
677247c0029SJamin Lin
678247c0029SJamin Lin set = &s->sets[set_idx];
679247c0029SJamin Lin props = &agc->props[set_idx];
680247c0029SJamin Lin
681247c0029SJamin Lin if (reg_idx_command)
682247c0029SJamin Lin qemu_log_mask(LOG_GUEST_ERROR, "%s: offset 0x%" PRIx64 "data 0x%"
683247c0029SJamin Lin PRIx64 "index mode wrong command 0x%x\n",
684247c0029SJamin Lin __func__, offset, data, reg_idx_command);
685247c0029SJamin Lin
686247c0029SJamin Lin switch (reg_idx_type) {
687247c0029SJamin Lin case gpio_reg_idx_data:
688247c0029SJamin Lin reg_value = set->data_read;
689247c0029SJamin Lin reg_value = deposit32(reg_value, pin_idx, 1,
690247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, DATA_VALUE));
691247c0029SJamin Lin reg_value &= props->output;
692247c0029SJamin Lin reg_value = update_value_control_source(set, set->data_value,
693247c0029SJamin Lin reg_value);
694247c0029SJamin Lin set->data_read = reg_value;
6951f30db92SPeter Delevoryas aspeed_gpio_update(s, set, reg_value, set->direction);
696247c0029SJamin Lin return;
697247c0029SJamin Lin case gpio_reg_idx_direction:
698247c0029SJamin Lin reg_value = set->direction;
699247c0029SJamin Lin reg_value = deposit32(reg_value, pin_idx, 1,
700247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, DIRECTION));
701247c0029SJamin Lin /*
702247c0029SJamin Lin * where data is the value attempted to be written to the pin:
703247c0029SJamin Lin * pin type | input mask | output mask | expected value
704247c0029SJamin Lin * ------------------------------------------------------------
705247c0029SJamin Lin * bidirectional | 1 | 1 | data
706247c0029SJamin Lin * input only | 1 | 0 | 0
707247c0029SJamin Lin * output only | 0 | 1 | 1
708247c0029SJamin Lin * no pin | 0 | 0 | 0
709247c0029SJamin Lin *
710247c0029SJamin Lin * which is captured by:
711247c0029SJamin Lin * data = ( data | ~input) & output;
712247c0029SJamin Lin */
713247c0029SJamin Lin reg_value = (reg_value | ~props->input) & props->output;
714247c0029SJamin Lin set->direction = update_value_control_source(set, set->direction,
715247c0029SJamin Lin reg_value);
716247c0029SJamin Lin break;
717247c0029SJamin Lin case gpio_reg_idx_interrupt:
718247c0029SJamin Lin reg_value = set->int_enable;
719247c0029SJamin Lin reg_value = deposit32(reg_value, pin_idx, 1,
720247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, INT_ENABLE));
721247c0029SJamin Lin set->int_enable = update_value_control_source(set, set->int_enable,
722247c0029SJamin Lin reg_value);
723247c0029SJamin Lin reg_value = set->int_sens_0;
724247c0029SJamin Lin reg_value = deposit32(reg_value, pin_idx, 1,
725247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, INT_SENS_0));
726247c0029SJamin Lin set->int_sens_0 = update_value_control_source(set, set->int_sens_0,
727247c0029SJamin Lin reg_value);
728247c0029SJamin Lin reg_value = set->int_sens_1;
729247c0029SJamin Lin reg_value = deposit32(reg_value, pin_idx, 1,
730247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, INT_SENS_1));
731247c0029SJamin Lin set->int_sens_1 = update_value_control_source(set, set->int_sens_1,
732247c0029SJamin Lin reg_value);
733247c0029SJamin Lin reg_value = set->int_sens_2;
734247c0029SJamin Lin reg_value = deposit32(reg_value, pin_idx, 1,
735247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, INT_SENS_2));
736247c0029SJamin Lin set->int_sens_2 = update_value_control_source(set, set->int_sens_2,
737247c0029SJamin Lin reg_value);
7387e22f6faSJamin Lin /* interrupt status */
7397e22f6faSJamin Lin if (FIELD_EX32(data, GPIO_INDEX_REG, INT_STATUS)) {
7407e22f6faSJamin Lin /* pending is either 1 or 0 for a 1-bit field */
7417e22f6faSJamin Lin pending = extract32(set->int_status, pin_idx, 1);
7427e22f6faSJamin Lin
7437e22f6faSJamin Lin assert(s->pending >= pending);
7447e22f6faSJamin Lin
7457e22f6faSJamin Lin /* No change to s->pending if pending is 0 */
7467e22f6faSJamin Lin s->pending -= pending;
7477e22f6faSJamin Lin
7487e22f6faSJamin Lin /*
7497e22f6faSJamin Lin * The write acknowledged the interrupt regardless of whether it
7507e22f6faSJamin Lin * was pending or not. The post-condition is that it mustn't be
7517e22f6faSJamin Lin * pending. Unconditionally clear the status bit.
7527e22f6faSJamin Lin */
7537e22f6faSJamin Lin set->int_status = deposit32(set->int_status, pin_idx, 1, 0);
754247c0029SJamin Lin }
755247c0029SJamin Lin break;
756247c0029SJamin Lin case gpio_reg_idx_debounce:
757247c0029SJamin Lin reg_value = set->debounce_1;
758247c0029SJamin Lin reg_value = deposit32(reg_value, pin_idx, 1,
759247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, DEBOUNCE_1));
760247c0029SJamin Lin set->debounce_1 = update_value_control_source(set, set->debounce_1,
761247c0029SJamin Lin reg_value);
762247c0029SJamin Lin reg_value = set->debounce_2;
763247c0029SJamin Lin reg_value = deposit32(reg_value, pin_idx, 1,
764247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, DEBOUNCE_2));
765247c0029SJamin Lin set->debounce_2 = update_value_control_source(set, set->debounce_2,
766247c0029SJamin Lin reg_value);
767247c0029SJamin Lin return;
768247c0029SJamin Lin case gpio_reg_idx_tolerance:
769247c0029SJamin Lin reg_value = set->reset_tol;
770247c0029SJamin Lin reg_value = deposit32(reg_value, pin_idx, 1,
771247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, RESET_TOLERANT));
772247c0029SJamin Lin set->reset_tol = update_value_control_source(set, set->reset_tol,
773247c0029SJamin Lin reg_value);
774247c0029SJamin Lin return;
775247c0029SJamin Lin case gpio_reg_idx_cmd_src:
776247c0029SJamin Lin reg_value = set->cmd_source_0;
777247c0029SJamin Lin reg_value = deposit32(reg_value, GPIOS_PER_GROUP * group_idx, 1,
778247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, COMMAND_SRC_0));
779247c0029SJamin Lin set->cmd_source_0 = reg_value & ASPEED_CMD_SRC_MASK;
780247c0029SJamin Lin reg_value = set->cmd_source_1;
781247c0029SJamin Lin reg_value = deposit32(reg_value, GPIOS_PER_GROUP * group_idx, 1,
782247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, COMMAND_SRC_1));
783247c0029SJamin Lin set->cmd_source_1 = reg_value & ASPEED_CMD_SRC_MASK;
784247c0029SJamin Lin return;
785247c0029SJamin Lin case gpio_reg_idx_input_mask:
786247c0029SJamin Lin reg_value = set->input_mask;
787247c0029SJamin Lin reg_value = deposit32(reg_value, pin_idx, 1,
788247c0029SJamin Lin FIELD_EX32(data, GPIO_INDEX_REG, INPUT_MASK));
789247c0029SJamin Lin /*
790247c0029SJamin Lin * feeds into interrupt generation
791247c0029SJamin Lin * 0: read from data value reg will be updated
792247c0029SJamin Lin * 1: read from data value reg will not be updated
793247c0029SJamin Lin */
794247c0029SJamin Lin set->input_mask = reg_value & props->input;
795247c0029SJamin Lin break;
796247c0029SJamin Lin default:
797247c0029SJamin Lin qemu_log_mask(LOG_GUEST_ERROR, "%s: offset 0x%" PRIx64 "data 0x%"
798247c0029SJamin Lin PRIx64 "index mode wrong type 0x%x\n",
799247c0029SJamin Lin __func__, offset, data, reg_idx_type);
800247c0029SJamin Lin return;
801247c0029SJamin Lin }
8021f30db92SPeter Delevoryas aspeed_gpio_update(s, set, set->data_value, UINT32_MAX);
803247c0029SJamin Lin return;
804247c0029SJamin Lin }
805247c0029SJamin Lin
aspeed_gpio_write(void * opaque,hwaddr offset,uint64_t data,uint32_t size)8064b7f9568SRashmica Gupta static void aspeed_gpio_write(void *opaque, hwaddr offset, uint64_t data,
8074b7f9568SRashmica Gupta uint32_t size)
8084b7f9568SRashmica Gupta {
8094b7f9568SRashmica Gupta AspeedGPIOState *s = ASPEED_GPIO(opaque);
8104b7f9568SRashmica Gupta AspeedGPIOClass *agc = ASPEED_GPIO_GET_CLASS(s);
8114b7f9568SRashmica Gupta const GPIOSetProperties *props;
8124b7f9568SRashmica Gupta uint64_t idx = -1;
8134b7f9568SRashmica Gupta const AspeedGPIOReg *reg;
8144b7f9568SRashmica Gupta GPIOSets *set;
8154b7f9568SRashmica Gupta uint32_t cleared;
8164b7f9568SRashmica Gupta
8177b1d21a8SJamin Lin trace_aspeed_gpio_write(offset, data);
8187b1d21a8SJamin Lin
8194b7f9568SRashmica Gupta idx = offset >> 2;
820247c0029SJamin Lin
821247c0029SJamin Lin /* check gpio index mode */
822247c0029SJamin Lin if (idx == R_GPIO_INDEX_REG) {
823247c0029SJamin Lin aspeed_gpio_write_index_mode(opaque, offset, data, size);
824247c0029SJamin Lin return;
825247c0029SJamin Lin }
826247c0029SJamin Lin
8274b7f9568SRashmica Gupta if (idx >= GPIO_DEBOUNCE_TIME_1 && idx <= GPIO_DEBOUNCE_TIME_3) {
8284b7f9568SRashmica Gupta idx -= GPIO_DEBOUNCE_TIME_1;
8294b7f9568SRashmica Gupta s->debounce_regs[idx] = (uint32_t) data;
8304b7f9568SRashmica Gupta return;
8314b7f9568SRashmica Gupta }
8324b7f9568SRashmica Gupta
83387511bb8SZheyu Ma if (idx >= agc->reg_table_count) {
83487511bb8SZheyu Ma qemu_log_mask(LOG_GUEST_ERROR, "%s: idx 0x%" PRIx64 " out of bounds\n",
83587511bb8SZheyu Ma __func__, idx);
83687511bb8SZheyu Ma return;
83787511bb8SZheyu Ma }
83887511bb8SZheyu Ma
8394b7f9568SRashmica Gupta reg = &agc->reg_table[idx];
8404b7f9568SRashmica Gupta if (reg->set_idx >= agc->nr_gpio_sets) {
8414b7f9568SRashmica Gupta qemu_log_mask(LOG_GUEST_ERROR, "%s: no setter for offset 0x%"
842554c2945SJamin Lin PRIx64"\n", __func__, offset);
8434b7f9568SRashmica Gupta return;
8444b7f9568SRashmica Gupta }
8454b7f9568SRashmica Gupta
8464b7f9568SRashmica Gupta set = &s->sets[reg->set_idx];
8474b7f9568SRashmica Gupta props = &agc->props[reg->set_idx];
8484b7f9568SRashmica Gupta
8494b7f9568SRashmica Gupta switch (reg->type) {
8504b7f9568SRashmica Gupta case gpio_reg_data_value:
8514b7f9568SRashmica Gupta data &= props->output;
8524b7f9568SRashmica Gupta data = update_value_control_source(set, set->data_value, data);
8534b7f9568SRashmica Gupta set->data_read = data;
8541f30db92SPeter Delevoryas aspeed_gpio_update(s, set, data, set->direction);
8554b7f9568SRashmica Gupta return;
8564b7f9568SRashmica Gupta case gpio_reg_direction:
8574b7f9568SRashmica Gupta /*
8584b7f9568SRashmica Gupta * where data is the value attempted to be written to the pin:
8594b7f9568SRashmica Gupta * pin type | input mask | output mask | expected value
8604b7f9568SRashmica Gupta * ------------------------------------------------------------
8614b7f9568SRashmica Gupta * bidirectional | 1 | 1 | data
8624b7f9568SRashmica Gupta * input only | 1 | 0 | 0
8634b7f9568SRashmica Gupta * output only | 0 | 1 | 1
86487bd33e8SPeter Delevoryas * no pin | 0 | 0 | 0
8654b7f9568SRashmica Gupta *
8664b7f9568SRashmica Gupta * which is captured by:
8674b7f9568SRashmica Gupta * data = ( data | ~input) & output;
8684b7f9568SRashmica Gupta */
8694b7f9568SRashmica Gupta data = (data | ~props->input) & props->output;
8704b7f9568SRashmica Gupta set->direction = update_value_control_source(set, set->direction, data);
8714b7f9568SRashmica Gupta break;
8724b7f9568SRashmica Gupta case gpio_reg_int_enable:
8734b7f9568SRashmica Gupta set->int_enable = update_value_control_source(set, set->int_enable,
8744b7f9568SRashmica Gupta data);
8754b7f9568SRashmica Gupta break;
8764b7f9568SRashmica Gupta case gpio_reg_int_sens_0:
8774b7f9568SRashmica Gupta set->int_sens_0 = update_value_control_source(set, set->int_sens_0,
8784b7f9568SRashmica Gupta data);
8794b7f9568SRashmica Gupta break;
8804b7f9568SRashmica Gupta case gpio_reg_int_sens_1:
8814b7f9568SRashmica Gupta set->int_sens_1 = update_value_control_source(set, set->int_sens_1,
8824b7f9568SRashmica Gupta data);
8834b7f9568SRashmica Gupta break;
8844b7f9568SRashmica Gupta case gpio_reg_int_sens_2:
8854b7f9568SRashmica Gupta set->int_sens_2 = update_value_control_source(set, set->int_sens_2,
8864b7f9568SRashmica Gupta data);
8874b7f9568SRashmica Gupta break;
8884b7f9568SRashmica Gupta case gpio_reg_int_status:
8894b7f9568SRashmica Gupta cleared = ctpop32(data & set->int_status);
8904b7f9568SRashmica Gupta if (s->pending && cleared) {
8914b7f9568SRashmica Gupta assert(s->pending >= cleared);
8924b7f9568SRashmica Gupta s->pending -= cleared;
8934b7f9568SRashmica Gupta }
8944b7f9568SRashmica Gupta set->int_status &= ~data;
8954b7f9568SRashmica Gupta break;
8964b7f9568SRashmica Gupta case gpio_reg_reset_tolerant:
8974b7f9568SRashmica Gupta set->reset_tol = update_value_control_source(set, set->reset_tol,
8984b7f9568SRashmica Gupta data);
8994b7f9568SRashmica Gupta return;
9004b7f9568SRashmica Gupta case gpio_reg_debounce_1:
9014b7f9568SRashmica Gupta set->debounce_1 = update_value_control_source(set, set->debounce_1,
9024b7f9568SRashmica Gupta data);
9034b7f9568SRashmica Gupta return;
9044b7f9568SRashmica Gupta case gpio_reg_debounce_2:
9054b7f9568SRashmica Gupta set->debounce_2 = update_value_control_source(set, set->debounce_2,
9064b7f9568SRashmica Gupta data);
9074b7f9568SRashmica Gupta return;
9084b7f9568SRashmica Gupta case gpio_reg_cmd_source_0:
9094b7f9568SRashmica Gupta set->cmd_source_0 = data & ASPEED_CMD_SRC_MASK;
9104b7f9568SRashmica Gupta return;
9114b7f9568SRashmica Gupta case gpio_reg_cmd_source_1:
9124b7f9568SRashmica Gupta set->cmd_source_1 = data & ASPEED_CMD_SRC_MASK;
9134b7f9568SRashmica Gupta return;
9144b7f9568SRashmica Gupta case gpio_reg_data_read:
9154b7f9568SRashmica Gupta /* Read only register */
9164b7f9568SRashmica Gupta return;
9174b7f9568SRashmica Gupta case gpio_reg_input_mask:
9184b7f9568SRashmica Gupta /*
9194b7f9568SRashmica Gupta * feeds into interrupt generation
9204b7f9568SRashmica Gupta * 0: read from data value reg will be updated
9214b7f9568SRashmica Gupta * 1: read from data value reg will not be updated
9224b7f9568SRashmica Gupta */
9234b7f9568SRashmica Gupta set->input_mask = data & props->input;
9244b7f9568SRashmica Gupta break;
9254b7f9568SRashmica Gupta default:
9264b7f9568SRashmica Gupta qemu_log_mask(LOG_GUEST_ERROR, "%s: no setter for offset 0x%"
927554c2945SJamin Lin PRIx64"\n", __func__, offset);
9284b7f9568SRashmica Gupta return;
9294b7f9568SRashmica Gupta }
9301f30db92SPeter Delevoryas aspeed_gpio_update(s, set, set->data_value, UINT32_MAX);
9314b7f9568SRashmica Gupta return;
9324b7f9568SRashmica Gupta }
9334b7f9568SRashmica Gupta
get_set_idx(AspeedGPIOState * s,const char * group,int * group_idx)9344b7f9568SRashmica Gupta static int get_set_idx(AspeedGPIOState *s, const char *group, int *group_idx)
9354b7f9568SRashmica Gupta {
9364b7f9568SRashmica Gupta AspeedGPIOClass *agc = ASPEED_GPIO_GET_CLASS(s);
9376ae1a5a3SPhilippe Mathieu-Daudé int set_idx, g_idx;
9384b7f9568SRashmica Gupta
9394b7f9568SRashmica Gupta for (set_idx = 0; set_idx < agc->nr_gpio_sets; set_idx++) {
9404b7f9568SRashmica Gupta const GPIOSetProperties *set_props = &agc->props[set_idx];
9414b7f9568SRashmica Gupta for (g_idx = 0; g_idx < ASPEED_GROUPS_PER_SET; g_idx++) {
9424b7f9568SRashmica Gupta if (!strncmp(group, set_props->group_label[g_idx], strlen(group))) {
9434b7f9568SRashmica Gupta *group_idx = g_idx;
9444b7f9568SRashmica Gupta return set_idx;
9454b7f9568SRashmica Gupta }
9464b7f9568SRashmica Gupta }
9474b7f9568SRashmica Gupta }
9484b7f9568SRashmica Gupta return -1;
9494b7f9568SRashmica Gupta }
9504b7f9568SRashmica Gupta
aspeed_gpio_get_pin(Object * obj,Visitor * v,const char * name,void * opaque,Error ** errp)9514b7f9568SRashmica Gupta static void aspeed_gpio_get_pin(Object *obj, Visitor *v, const char *name,
9524b7f9568SRashmica Gupta void *opaque, Error **errp)
9534b7f9568SRashmica Gupta {
9544b7f9568SRashmica Gupta int pin = 0xfff;
9554b7f9568SRashmica Gupta bool level = true;
9567811ce81SCédric Le Goater char group[4];
9574b7f9568SRashmica Gupta AspeedGPIOState *s = ASPEED_GPIO(obj);
9584b7f9568SRashmica Gupta int set_idx, group_idx = 0;
9594b7f9568SRashmica Gupta
9604b7f9568SRashmica Gupta if (sscanf(name, "gpio%2[A-Z]%1d", group, &pin) != 2) {
96136d737eeSRashmica Gupta /* 1.8V gpio */
9627811ce81SCédric Le Goater if (sscanf(name, "gpio%3[18A-E]%1d", group, &pin) != 2) {
9634b7f9568SRashmica Gupta error_setg(errp, "%s: error reading %s", __func__, name);
9644b7f9568SRashmica Gupta return;
9654b7f9568SRashmica Gupta }
96636d737eeSRashmica Gupta }
9674b7f9568SRashmica Gupta set_idx = get_set_idx(s, group, &group_idx);
9684b7f9568SRashmica Gupta if (set_idx == -1) {
9694b7f9568SRashmica Gupta error_setg(errp, "%s: invalid group %s", __func__, group);
9704b7f9568SRashmica Gupta return;
9714b7f9568SRashmica Gupta }
9724b7f9568SRashmica Gupta pin = pin + group_idx * GPIOS_PER_GROUP;
9734b7f9568SRashmica Gupta level = aspeed_gpio_get_pin_level(s, set_idx, pin);
9744b7f9568SRashmica Gupta visit_type_bool(v, name, &level, errp);
9754b7f9568SRashmica Gupta }
9764b7f9568SRashmica Gupta
aspeed_gpio_set_pin(Object * obj,Visitor * v,const char * name,void * opaque,Error ** errp)9774b7f9568SRashmica Gupta static void aspeed_gpio_set_pin(Object *obj, Visitor *v, const char *name,
9784b7f9568SRashmica Gupta void *opaque, Error **errp)
9794b7f9568SRashmica Gupta {
9804b7f9568SRashmica Gupta bool level;
9814b7f9568SRashmica Gupta int pin = 0xfff;
9827811ce81SCédric Le Goater char group[4];
9834b7f9568SRashmica Gupta AspeedGPIOState *s = ASPEED_GPIO(obj);
9844b7f9568SRashmica Gupta int set_idx, group_idx = 0;
9854b7f9568SRashmica Gupta
986668f62ecSMarkus Armbruster if (!visit_type_bool(v, name, &level, errp)) {
9874b7f9568SRashmica Gupta return;
9884b7f9568SRashmica Gupta }
9894b7f9568SRashmica Gupta if (sscanf(name, "gpio%2[A-Z]%1d", group, &pin) != 2) {
99036d737eeSRashmica Gupta /* 1.8V gpio */
9917811ce81SCédric Le Goater if (sscanf(name, "gpio%3[18A-E]%1d", group, &pin) != 2) {
9924b7f9568SRashmica Gupta error_setg(errp, "%s: error reading %s", __func__, name);
9934b7f9568SRashmica Gupta return;
9944b7f9568SRashmica Gupta }
99536d737eeSRashmica Gupta }
9964b7f9568SRashmica Gupta set_idx = get_set_idx(s, group, &group_idx);
9974b7f9568SRashmica Gupta if (set_idx == -1) {
9984b7f9568SRashmica Gupta error_setg(errp, "%s: invalid group %s", __func__, group);
9994b7f9568SRashmica Gupta return;
10004b7f9568SRashmica Gupta }
10014b7f9568SRashmica Gupta pin = pin + group_idx * GPIOS_PER_GROUP;
10024b7f9568SRashmica Gupta aspeed_gpio_set_pin_level(s, set_idx, pin, level);
10034b7f9568SRashmica Gupta }
10044b7f9568SRashmica Gupta
aspeed_gpio_2700_read_control_reg(AspeedGPIOState * s,uint32_t pin)1005bac69883SJamin Lin static uint64_t aspeed_gpio_2700_read_control_reg(AspeedGPIOState *s,
1006bac69883SJamin Lin uint32_t pin)
1007bac69883SJamin Lin {
1008bac69883SJamin Lin AspeedGPIOClass *agc = ASPEED_GPIO_GET_CLASS(s);
1009bac69883SJamin Lin GPIOSets *set;
1010bac69883SJamin Lin uint64_t value = 0;
1011bac69883SJamin Lin uint32_t set_idx;
1012bac69883SJamin Lin uint32_t pin_idx;
1013bac69883SJamin Lin
1014bac69883SJamin Lin set_idx = pin / ASPEED_GPIOS_PER_SET;
1015bac69883SJamin Lin pin_idx = pin % ASPEED_GPIOS_PER_SET;
1016bac69883SJamin Lin
1017bac69883SJamin Lin if (set_idx >= agc->nr_gpio_sets) {
1018bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR, "%s: set index: %d, out of bounds\n",
1019bac69883SJamin Lin __func__, set_idx);
1020bac69883SJamin Lin return 0;
1021bac69883SJamin Lin }
1022bac69883SJamin Lin
1023bac69883SJamin Lin set = &s->sets[set_idx];
1024bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_OUT_DATA,
1025bac69883SJamin Lin extract32(set->data_read, pin_idx, 1));
1026bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_DIRECTION,
1027bac69883SJamin Lin extract32(set->direction, pin_idx, 1));
1028bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_INT_ENABLE,
1029bac69883SJamin Lin extract32(set->int_enable, pin_idx, 1));
1030bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_INT_SENS_0,
1031bac69883SJamin Lin extract32(set->int_sens_0, pin_idx, 1));
1032bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_INT_SENS_1,
1033bac69883SJamin Lin extract32(set->int_sens_1, pin_idx, 1));
1034bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_INT_SENS_2,
1035bac69883SJamin Lin extract32(set->int_sens_2, pin_idx, 1));
1036bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_RESET_TOLERANCE,
1037bac69883SJamin Lin extract32(set->reset_tol, pin_idx, 1));
1038bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_DEBOUNCE_1,
1039bac69883SJamin Lin extract32(set->debounce_1, pin_idx, 1));
1040bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_DEBOUNCE_2,
1041bac69883SJamin Lin extract32(set->debounce_2, pin_idx, 1));
1042bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_INPUT_MASK,
1043bac69883SJamin Lin extract32(set->input_mask, pin_idx, 1));
1044bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_INT_STATUS,
1045bac69883SJamin Lin extract32(set->int_status, pin_idx, 1));
1046bac69883SJamin Lin value = SHARED_FIELD_DP32(value, GPIO_CONTROL_IN_DATA,
1047bac69883SJamin Lin extract32(set->data_value, pin_idx, 1));
1048bac69883SJamin Lin return value;
1049bac69883SJamin Lin }
1050bac69883SJamin Lin
aspeed_gpio_2700_write_control_reg(AspeedGPIOState * s,uint32_t pin,uint64_t data)1051bac69883SJamin Lin static void aspeed_gpio_2700_write_control_reg(AspeedGPIOState *s,
1052bac69883SJamin Lin uint32_t pin, uint64_t data)
1053bac69883SJamin Lin {
1054bac69883SJamin Lin AspeedGPIOClass *agc = ASPEED_GPIO_GET_CLASS(s);
1055bac69883SJamin Lin const GPIOSetProperties *props;
1056bac69883SJamin Lin GPIOSets *set;
1057bac69883SJamin Lin uint32_t set_idx;
1058bac69883SJamin Lin uint32_t pin_idx;
1059bac69883SJamin Lin uint32_t group_value = 0;
1060bac69883SJamin Lin uint32_t pending = 0;
1061bac69883SJamin Lin
1062bac69883SJamin Lin set_idx = pin / ASPEED_GPIOS_PER_SET;
1063bac69883SJamin Lin pin_idx = pin % ASPEED_GPIOS_PER_SET;
1064bac69883SJamin Lin
1065bac69883SJamin Lin if (set_idx >= agc->nr_gpio_sets) {
1066bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR, "%s: set index: %d, out of bounds\n",
1067bac69883SJamin Lin __func__, set_idx);
1068bac69883SJamin Lin return;
1069bac69883SJamin Lin }
1070bac69883SJamin Lin
1071bac69883SJamin Lin set = &s->sets[set_idx];
1072bac69883SJamin Lin props = &agc->props[set_idx];
1073bac69883SJamin Lin
1074bac69883SJamin Lin /* direction */
1075bac69883SJamin Lin group_value = set->direction;
1076bac69883SJamin Lin group_value = deposit32(group_value, pin_idx, 1,
1077bac69883SJamin Lin SHARED_FIELD_EX32(data, GPIO_CONTROL_DIRECTION));
1078bac69883SJamin Lin /*
1079bac69883SJamin Lin * where data is the value attempted to be written to the pin:
1080bac69883SJamin Lin * pin type | input mask | output mask | expected value
1081bac69883SJamin Lin * ------------------------------------------------------------
1082bac69883SJamin Lin * bidirectional | 1 | 1 | data
1083bac69883SJamin Lin * input only | 1 | 0 | 0
1084bac69883SJamin Lin * output only | 0 | 1 | 1
1085bac69883SJamin Lin * no pin | 0 | 0 | 0
1086bac69883SJamin Lin *
1087bac69883SJamin Lin * which is captured by:
1088bac69883SJamin Lin * data = ( data | ~input) & output;
1089bac69883SJamin Lin */
1090bac69883SJamin Lin group_value = (group_value | ~props->input) & props->output;
1091bac69883SJamin Lin set->direction = update_value_control_source(set, set->direction,
1092bac69883SJamin Lin group_value);
1093bac69883SJamin Lin
1094bac69883SJamin Lin /* out data */
1095bac69883SJamin Lin group_value = set->data_read;
1096bac69883SJamin Lin group_value = deposit32(group_value, pin_idx, 1,
1097bac69883SJamin Lin SHARED_FIELD_EX32(data, GPIO_CONTROL_OUT_DATA));
1098bac69883SJamin Lin group_value &= props->output;
1099bac69883SJamin Lin group_value = update_value_control_source(set, set->data_read,
1100bac69883SJamin Lin group_value);
1101bac69883SJamin Lin set->data_read = group_value;
1102bac69883SJamin Lin
1103bac69883SJamin Lin /* interrupt enable */
1104bac69883SJamin Lin group_value = set->int_enable;
1105bac69883SJamin Lin group_value = deposit32(group_value, pin_idx, 1,
1106bac69883SJamin Lin SHARED_FIELD_EX32(data, GPIO_CONTROL_INT_ENABLE));
1107bac69883SJamin Lin set->int_enable = update_value_control_source(set, set->int_enable,
1108bac69883SJamin Lin group_value);
1109bac69883SJamin Lin
1110bac69883SJamin Lin /* interrupt sensitivity type 0 */
1111bac69883SJamin Lin group_value = set->int_sens_0;
1112bac69883SJamin Lin group_value = deposit32(group_value, pin_idx, 1,
1113bac69883SJamin Lin SHARED_FIELD_EX32(data, GPIO_CONTROL_INT_SENS_0));
1114bac69883SJamin Lin set->int_sens_0 = update_value_control_source(set, set->int_sens_0,
1115bac69883SJamin Lin group_value);
1116bac69883SJamin Lin
1117bac69883SJamin Lin /* interrupt sensitivity type 1 */
1118bac69883SJamin Lin group_value = set->int_sens_1;
1119bac69883SJamin Lin group_value = deposit32(group_value, pin_idx, 1,
1120bac69883SJamin Lin SHARED_FIELD_EX32(data, GPIO_CONTROL_INT_SENS_1));
1121bac69883SJamin Lin set->int_sens_1 = update_value_control_source(set, set->int_sens_1,
1122bac69883SJamin Lin group_value);
1123bac69883SJamin Lin
1124bac69883SJamin Lin /* interrupt sensitivity type 2 */
1125bac69883SJamin Lin group_value = set->int_sens_2;
1126bac69883SJamin Lin group_value = deposit32(group_value, pin_idx, 1,
1127bac69883SJamin Lin SHARED_FIELD_EX32(data, GPIO_CONTROL_INT_SENS_2));
1128bac69883SJamin Lin set->int_sens_2 = update_value_control_source(set, set->int_sens_2,
1129bac69883SJamin Lin group_value);
1130bac69883SJamin Lin
1131bac69883SJamin Lin /* reset tolerance enable */
1132bac69883SJamin Lin group_value = set->reset_tol;
1133bac69883SJamin Lin group_value = deposit32(group_value, pin_idx, 1,
1134bac69883SJamin Lin SHARED_FIELD_EX32(data, GPIO_CONTROL_RESET_TOLERANCE));
1135bac69883SJamin Lin set->reset_tol = update_value_control_source(set, set->reset_tol,
1136bac69883SJamin Lin group_value);
1137bac69883SJamin Lin
1138bac69883SJamin Lin /* debounce 1 */
1139bac69883SJamin Lin group_value = set->debounce_1;
1140bac69883SJamin Lin group_value = deposit32(group_value, pin_idx, 1,
1141bac69883SJamin Lin SHARED_FIELD_EX32(data, GPIO_CONTROL_DEBOUNCE_1));
1142bac69883SJamin Lin set->debounce_1 = update_value_control_source(set, set->debounce_1,
1143bac69883SJamin Lin group_value);
1144bac69883SJamin Lin
1145bac69883SJamin Lin /* debounce 2 */
1146bac69883SJamin Lin group_value = set->debounce_2;
1147bac69883SJamin Lin group_value = deposit32(group_value, pin_idx, 1,
1148bac69883SJamin Lin SHARED_FIELD_EX32(data, GPIO_CONTROL_DEBOUNCE_2));
1149bac69883SJamin Lin set->debounce_2 = update_value_control_source(set, set->debounce_2,
1150bac69883SJamin Lin group_value);
1151bac69883SJamin Lin
1152bac69883SJamin Lin /* input mask */
1153bac69883SJamin Lin group_value = set->input_mask;
1154bac69883SJamin Lin group_value = deposit32(group_value, pin_idx, 1,
1155bac69883SJamin Lin SHARED_FIELD_EX32(data, GPIO_CONTROL_INPUT_MASK));
1156bac69883SJamin Lin /*
1157bac69883SJamin Lin * feeds into interrupt generation
1158bac69883SJamin Lin * 0: read from data value reg will be updated
1159bac69883SJamin Lin * 1: read from data value reg will not be updated
1160bac69883SJamin Lin */
1161bac69883SJamin Lin set->input_mask = group_value & props->input;
1162bac69883SJamin Lin
1163bac69883SJamin Lin /* blink counter 1 */
1164bac69883SJamin Lin /* blink counter 2 */
1165bac69883SJamin Lin /* unimplement */
1166bac69883SJamin Lin
1167bac69883SJamin Lin /* interrupt status */
1168bac69883SJamin Lin if (SHARED_FIELD_EX32(data, GPIO_CONTROL_INT_STATUS)) {
1169bac69883SJamin Lin /* pending is either 1 or 0 for a 1-bit field */
1170bac69883SJamin Lin pending = extract32(set->int_status, pin_idx, 1);
1171bac69883SJamin Lin
1172bac69883SJamin Lin assert(s->pending >= pending);
1173bac69883SJamin Lin
1174bac69883SJamin Lin /* No change to s->pending if pending is 0 */
1175bac69883SJamin Lin s->pending -= pending;
1176bac69883SJamin Lin
1177bac69883SJamin Lin /*
1178bac69883SJamin Lin * The write acknowledged the interrupt regardless of whether it
1179bac69883SJamin Lin * was pending or not. The post-condition is that it mustn't be
1180bac69883SJamin Lin * pending. Unconditionally clear the status bit.
1181bac69883SJamin Lin */
1182bac69883SJamin Lin set->int_status = deposit32(set->int_status, pin_idx, 1, 0);
1183bac69883SJamin Lin }
1184bac69883SJamin Lin
1185bac69883SJamin Lin aspeed_gpio_update(s, set, set->data_value, UINT32_MAX);
1186bac69883SJamin Lin return;
1187bac69883SJamin Lin }
1188bac69883SJamin Lin
aspeed_gpio_2700_read(void * opaque,hwaddr offset,uint32_t size)1189bac69883SJamin Lin static uint64_t aspeed_gpio_2700_read(void *opaque, hwaddr offset,
1190bac69883SJamin Lin uint32_t size)
1191bac69883SJamin Lin {
1192bac69883SJamin Lin AspeedGPIOState *s = ASPEED_GPIO(opaque);
1193bac69883SJamin Lin AspeedGPIOClass *agc = ASPEED_GPIO_GET_CLASS(s);
1194bac69883SJamin Lin GPIOSets *set;
1195bac69883SJamin Lin uint64_t value;
1196bac69883SJamin Lin uint64_t reg;
1197bac69883SJamin Lin uint32_t pin;
1198bac69883SJamin Lin uint32_t idx;
1199bac69883SJamin Lin
1200bac69883SJamin Lin reg = offset >> 2;
1201bac69883SJamin Lin
1202bac69883SJamin Lin if (reg >= agc->reg_table_count) {
1203bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR,
1204bac69883SJamin Lin "%s: offset 0x%" PRIx64 " out of bounds\n",
1205bac69883SJamin Lin __func__, offset);
1206bac69883SJamin Lin return 0;
1207bac69883SJamin Lin }
1208bac69883SJamin Lin
1209bac69883SJamin Lin switch (reg) {
1210bac69883SJamin Lin case R_GPIO_2700_DEBOUNCE_TIME_1 ... R_GPIO_2700_DEBOUNCE_TIME_3:
1211bac69883SJamin Lin idx = reg - R_GPIO_2700_DEBOUNCE_TIME_1;
1212bac69883SJamin Lin
1213bac69883SJamin Lin if (idx >= ASPEED_GPIO_NR_DEBOUNCE_REGS) {
1214bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR,
1215bac69883SJamin Lin "%s: debounce index: %d, out of bounds\n",
1216bac69883SJamin Lin __func__, idx);
1217bac69883SJamin Lin return 0;
1218bac69883SJamin Lin }
1219bac69883SJamin Lin
1220bac69883SJamin Lin value = (uint64_t) s->debounce_regs[idx];
1221bac69883SJamin Lin break;
1222bac69883SJamin Lin case R_GPIO_2700_INT_STATUS_1 ... R_GPIO_2700_INT_STATUS_7:
1223bac69883SJamin Lin idx = reg - R_GPIO_2700_INT_STATUS_1;
1224bac69883SJamin Lin
1225bac69883SJamin Lin if (idx >= agc->nr_gpio_sets) {
1226bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR,
1227bac69883SJamin Lin "%s: interrupt status index: %d, out of bounds\n",
1228bac69883SJamin Lin __func__, idx);
1229bac69883SJamin Lin return 0;
1230bac69883SJamin Lin }
1231bac69883SJamin Lin
1232bac69883SJamin Lin set = &s->sets[idx];
1233bac69883SJamin Lin value = (uint64_t) set->int_status;
1234bac69883SJamin Lin break;
1235bac69883SJamin Lin case R_GPIO_A0_CONTROL ... R_GPIO_AA7_CONTROL:
1236bac69883SJamin Lin pin = reg - R_GPIO_A0_CONTROL;
1237bac69883SJamin Lin
1238bac69883SJamin Lin if (pin >= agc->nr_gpio_pins) {
1239bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR, "%s: invalid pin number: %d\n",
1240bac69883SJamin Lin __func__, pin);
1241bac69883SJamin Lin return 0;
1242bac69883SJamin Lin }
1243bac69883SJamin Lin
1244bac69883SJamin Lin value = aspeed_gpio_2700_read_control_reg(s, pin);
1245bac69883SJamin Lin break;
1246bac69883SJamin Lin default:
1247bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR, "%s: no getter for offset 0x%"
1248bac69883SJamin Lin PRIx64"\n", __func__, offset);
1249bac69883SJamin Lin return 0;
1250bac69883SJamin Lin }
1251bac69883SJamin Lin
1252bac69883SJamin Lin trace_aspeed_gpio_read(offset, value);
1253bac69883SJamin Lin return value;
1254bac69883SJamin Lin }
1255bac69883SJamin Lin
aspeed_gpio_2700_write(void * opaque,hwaddr offset,uint64_t data,uint32_t size)1256bac69883SJamin Lin static void aspeed_gpio_2700_write(void *opaque, hwaddr offset,
1257bac69883SJamin Lin uint64_t data, uint32_t size)
1258bac69883SJamin Lin {
1259bac69883SJamin Lin AspeedGPIOState *s = ASPEED_GPIO(opaque);
1260bac69883SJamin Lin AspeedGPIOClass *agc = ASPEED_GPIO_GET_CLASS(s);
1261bac69883SJamin Lin uint64_t reg;
1262bac69883SJamin Lin uint32_t pin;
1263bac69883SJamin Lin uint32_t idx;
1264bac69883SJamin Lin
1265bac69883SJamin Lin trace_aspeed_gpio_write(offset, data);
1266bac69883SJamin Lin
1267bac69883SJamin Lin reg = offset >> 2;
1268bac69883SJamin Lin
1269bac69883SJamin Lin if (reg >= agc->reg_table_count) {
1270bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR,
1271bac69883SJamin Lin "%s: offset 0x%" PRIx64 " out of bounds\n",
1272bac69883SJamin Lin __func__, offset);
1273bac69883SJamin Lin return;
1274bac69883SJamin Lin }
1275bac69883SJamin Lin
1276bac69883SJamin Lin switch (reg) {
1277bac69883SJamin Lin case R_GPIO_2700_DEBOUNCE_TIME_1 ... R_GPIO_2700_DEBOUNCE_TIME_3:
1278bac69883SJamin Lin idx = reg - R_GPIO_2700_DEBOUNCE_TIME_1;
1279bac69883SJamin Lin
1280bac69883SJamin Lin if (idx >= ASPEED_GPIO_NR_DEBOUNCE_REGS) {
1281bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR,
1282bac69883SJamin Lin "%s: debounce index: %d out of bounds\n",
1283bac69883SJamin Lin __func__, idx);
1284bac69883SJamin Lin return;
1285bac69883SJamin Lin }
1286bac69883SJamin Lin
1287bac69883SJamin Lin s->debounce_regs[idx] = (uint32_t) data;
1288bac69883SJamin Lin break;
1289bac69883SJamin Lin case R_GPIO_A0_CONTROL ... R_GPIO_AA7_CONTROL:
1290bac69883SJamin Lin pin = reg - R_GPIO_A0_CONTROL;
1291bac69883SJamin Lin
1292bac69883SJamin Lin if (pin >= agc->nr_gpio_pins) {
1293bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR, "%s: invalid pin number: %d\n",
1294bac69883SJamin Lin __func__, pin);
1295bac69883SJamin Lin return;
1296bac69883SJamin Lin }
1297bac69883SJamin Lin
1298bac69883SJamin Lin if (SHARED_FIELD_EX32(data, GPIO_CONTROL_RESERVED)) {
1299bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR, "%s: invalid reserved data: 0x%"
1300bac69883SJamin Lin PRIx64"\n", __func__, data);
1301bac69883SJamin Lin return;
1302bac69883SJamin Lin }
1303bac69883SJamin Lin
1304bac69883SJamin Lin aspeed_gpio_2700_write_control_reg(s, pin, data);
1305bac69883SJamin Lin break;
1306bac69883SJamin Lin default:
1307bac69883SJamin Lin qemu_log_mask(LOG_GUEST_ERROR, "%s: no setter for offset 0x%"
1308bac69883SJamin Lin PRIx64"\n", __func__, offset);
1309bac69883SJamin Lin break;
1310bac69883SJamin Lin }
1311bac69883SJamin Lin
1312bac69883SJamin Lin return;
1313bac69883SJamin Lin }
1314bac69883SJamin Lin
131533343bffSJamin Lin /* Setup functions */
131687bd33e8SPeter Delevoryas static const GPIOSetProperties ast2400_set_props[ASPEED_GPIO_MAX_NR_SETS] = {
13174b7f9568SRashmica Gupta [0] = {0xffffffff, 0xffffffff, {"A", "B", "C", "D"} },
13184b7f9568SRashmica Gupta [1] = {0xffffffff, 0xffffffff, {"E", "F", "G", "H"} },
13194b7f9568SRashmica Gupta [2] = {0xffffffff, 0xffffffff, {"I", "J", "K", "L"} },
13204b7f9568SRashmica Gupta [3] = {0xffffffff, 0xffffffff, {"M", "N", "O", "P"} },
13214b7f9568SRashmica Gupta [4] = {0xffffffff, 0xffffffff, {"Q", "R", "S", "T"} },
13224b7f9568SRashmica Gupta [5] = {0xffffffff, 0x0000ffff, {"U", "V", "W", "X"} },
13234b7f9568SRashmica Gupta [6] = {0x0000000f, 0x0fffff0f, {"Y", "Z", "AA", "AB"} },
13244b7f9568SRashmica Gupta };
13254b7f9568SRashmica Gupta
132687bd33e8SPeter Delevoryas static const GPIOSetProperties ast2500_set_props[ASPEED_GPIO_MAX_NR_SETS] = {
13274b7f9568SRashmica Gupta [0] = {0xffffffff, 0xffffffff, {"A", "B", "C", "D"} },
13284b7f9568SRashmica Gupta [1] = {0xffffffff, 0xffffffff, {"E", "F", "G", "H"} },
13294b7f9568SRashmica Gupta [2] = {0xffffffff, 0xffffffff, {"I", "J", "K", "L"} },
13304b7f9568SRashmica Gupta [3] = {0xffffffff, 0xffffffff, {"M", "N", "O", "P"} },
13314b7f9568SRashmica Gupta [4] = {0xffffffff, 0xffffffff, {"Q", "R", "S", "T"} },
13324b7f9568SRashmica Gupta [5] = {0xffffffff, 0x0000ffff, {"U", "V", "W", "X"} },
13339fffe140SPeter Delevoryas [6] = {0x0fffffff, 0x0fffffff, {"Y", "Z", "AA", "AB"} },
13344b7f9568SRashmica Gupta [7] = {0x000000ff, 0x000000ff, {"AC"} },
13354b7f9568SRashmica Gupta };
13364b7f9568SRashmica Gupta
133787bd33e8SPeter Delevoryas static GPIOSetProperties ast2600_3_3v_set_props[ASPEED_GPIO_MAX_NR_SETS] = {
133836d737eeSRashmica Gupta [0] = {0xffffffff, 0xffffffff, {"A", "B", "C", "D"} },
133936d737eeSRashmica Gupta [1] = {0xffffffff, 0xffffffff, {"E", "F", "G", "H"} },
134036d737eeSRashmica Gupta [2] = {0xffffffff, 0xffffffff, {"I", "J", "K", "L"} },
134136d737eeSRashmica Gupta [3] = {0xffffffff, 0xffffffff, {"M", "N", "O", "P"} },
13429fffe140SPeter Delevoryas [4] = {0xffffffff, 0x00ffffff, {"Q", "R", "S", "T"} },
13439fffe140SPeter Delevoryas [5] = {0xffffffff, 0xffffff00, {"U", "V", "W", "X"} },
13449fffe140SPeter Delevoryas [6] = {0x0000ffff, 0x0000ffff, {"Y", "Z"} },
134536d737eeSRashmica Gupta };
134636d737eeSRashmica Gupta
134787bd33e8SPeter Delevoryas static GPIOSetProperties ast2600_1_8v_set_props[ASPEED_GPIO_MAX_NR_SETS] = {
134836d737eeSRashmica Gupta [0] = {0xffffffff, 0xffffffff, {"18A", "18B", "18C", "18D"} },
134936d737eeSRashmica Gupta [1] = {0x0000000f, 0x0000000f, {"18E"} },
135036d737eeSRashmica Gupta };
135136d737eeSRashmica Gupta
135217075ef2SJamin Lin static GPIOSetProperties ast1030_set_props[ASPEED_GPIO_MAX_NR_SETS] = {
135317075ef2SJamin Lin [0] = {0xffffffff, 0xffffffff, {"A", "B", "C", "D"} },
135417075ef2SJamin Lin [1] = {0xffffffff, 0xffffffff, {"E", "F", "G", "H"} },
135517075ef2SJamin Lin [2] = {0xffffffff, 0xffffffff, {"I", "J", "K", "L"} },
135617075ef2SJamin Lin [3] = {0xffffff3f, 0xffffff3f, {"M", "N", "O", "P"} },
135717075ef2SJamin Lin [4] = {0xff060c1f, 0x00060c1f, {"Q", "R", "S", "T"} },
135817075ef2SJamin Lin [5] = {0x000000ff, 0x00000000, {"U"} },
135917075ef2SJamin Lin };
136017075ef2SJamin Lin
1361bac69883SJamin Lin static GPIOSetProperties ast2700_set_props[ASPEED_GPIO_MAX_NR_SETS] = {
1362bac69883SJamin Lin [0] = {0xffffffff, 0xffffffff, {"A", "B", "C", "D"} },
1363bac69883SJamin Lin [1] = {0x0fffffff, 0x0fffffff, {"E", "F", "G", "H"} },
1364bac69883SJamin Lin [2] = {0xffffffff, 0xffffffff, {"I", "J", "K", "L"} },
1365bac69883SJamin Lin [3] = {0xffffffff, 0xffffffff, {"M", "N", "O", "P"} },
1366bac69883SJamin Lin [4] = {0xffffffff, 0xffffffff, {"Q", "R", "S", "T"} },
1367bac69883SJamin Lin [5] = {0xffffffff, 0xffffffff, {"U", "V", "W", "X"} },
1368bac69883SJamin Lin [6] = {0x00ffffff, 0x00ffffff, {"Y", "Z", "AA"} },
1369bac69883SJamin Lin };
1370bac69883SJamin Lin
13714b7f9568SRashmica Gupta static const MemoryRegionOps aspeed_gpio_ops = {
13724b7f9568SRashmica Gupta .read = aspeed_gpio_read,
13734b7f9568SRashmica Gupta .write = aspeed_gpio_write,
13744b7f9568SRashmica Gupta .endianness = DEVICE_LITTLE_ENDIAN,
1375*d34fea6cSJoel Stanley .valid.min_access_size = 1,
13764b7f9568SRashmica Gupta .valid.max_access_size = 4,
13774b7f9568SRashmica Gupta };
13784b7f9568SRashmica Gupta
1379bac69883SJamin Lin static const MemoryRegionOps aspeed_gpio_2700_ops = {
1380bac69883SJamin Lin .read = aspeed_gpio_2700_read,
1381bac69883SJamin Lin .write = aspeed_gpio_2700_write,
1382bac69883SJamin Lin .endianness = DEVICE_LITTLE_ENDIAN,
1383*d34fea6cSJoel Stanley .valid.min_access_size = 1,
1384bac69883SJamin Lin .valid.max_access_size = 4,
1385bac69883SJamin Lin };
1386bac69883SJamin Lin
aspeed_gpio_reset(DeviceState * dev)13874b7f9568SRashmica Gupta static void aspeed_gpio_reset(DeviceState *dev)
13884b7f9568SRashmica Gupta {
13894b7f9568SRashmica Gupta AspeedGPIOState *s = ASPEED_GPIO(dev);
13904b7f9568SRashmica Gupta
13914b7f9568SRashmica Gupta /* TODO: respect the reset tolerance registers */
13924b7f9568SRashmica Gupta memset(s->sets, 0, sizeof(s->sets));
13934b7f9568SRashmica Gupta }
13944b7f9568SRashmica Gupta
aspeed_gpio_realize(DeviceState * dev,Error ** errp)13954b7f9568SRashmica Gupta static void aspeed_gpio_realize(DeviceState *dev, Error **errp)
13964b7f9568SRashmica Gupta {
13974b7f9568SRashmica Gupta AspeedGPIOState *s = ASPEED_GPIO(dev);
13984b7f9568SRashmica Gupta SysBusDevice *sbd = SYS_BUS_DEVICE(dev);
13994b7f9568SRashmica Gupta AspeedGPIOClass *agc = ASPEED_GPIO_GET_CLASS(s);
14004b7f9568SRashmica Gupta
14014b7f9568SRashmica Gupta /* Interrupt parent line */
14024b7f9568SRashmica Gupta sysbus_init_irq(sbd, &s->irq);
14034b7f9568SRashmica Gupta
14044b7f9568SRashmica Gupta /* Individual GPIOs */
140587bd33e8SPeter Delevoryas for (int i = 0; i < ASPEED_GPIO_MAX_NR_SETS; i++) {
140687bd33e8SPeter Delevoryas const GPIOSetProperties *props = &agc->props[i];
140787bd33e8SPeter Delevoryas uint32_t skip = ~(props->input | props->output);
140887bd33e8SPeter Delevoryas for (int j = 0; j < ASPEED_GPIOS_PER_SET; j++) {
140987bd33e8SPeter Delevoryas if (skip >> j & 1) {
141087bd33e8SPeter Delevoryas continue;
141187bd33e8SPeter Delevoryas }
141287bd33e8SPeter Delevoryas sysbus_init_irq(sbd, &s->gpios[i][j]);
141387bd33e8SPeter Delevoryas }
14144b7f9568SRashmica Gupta }
14154b7f9568SRashmica Gupta
1416404e7534SJamin Lin memory_region_init_io(&s->iomem, OBJECT(s), agc->reg_ops, s,
14179422dbd1SJamin Lin TYPE_ASPEED_GPIO, agc->mem_size);
14184b7f9568SRashmica Gupta
14194b7f9568SRashmica Gupta sysbus_init_mmio(sbd, &s->iomem);
14204b7f9568SRashmica Gupta }
14214b7f9568SRashmica Gupta
aspeed_gpio_init(Object * obj)14224b7f9568SRashmica Gupta static void aspeed_gpio_init(Object *obj)
14234b7f9568SRashmica Gupta {
14244b7f9568SRashmica Gupta AspeedGPIOState *s = ASPEED_GPIO(obj);
14254b7f9568SRashmica Gupta AspeedGPIOClass *agc = ASPEED_GPIO_GET_CLASS(s);
14264b7f9568SRashmica Gupta
142787bd33e8SPeter Delevoryas for (int i = 0; i < ASPEED_GPIO_MAX_NR_SETS; i++) {
142887bd33e8SPeter Delevoryas const GPIOSetProperties *props = &agc->props[i];
142987bd33e8SPeter Delevoryas uint32_t skip = ~(props->input | props->output);
143087bd33e8SPeter Delevoryas for (int j = 0; j < ASPEED_GPIOS_PER_SET; j++) {
143187bd33e8SPeter Delevoryas if (skip >> j & 1) {
143287bd33e8SPeter Delevoryas continue;
143387bd33e8SPeter Delevoryas }
143487bd33e8SPeter Delevoryas int group_idx = j / GPIOS_PER_GROUP;
143587bd33e8SPeter Delevoryas int pin_idx = j % GPIOS_PER_GROUP;
143687bd33e8SPeter Delevoryas const char *group = &props->group_label[group_idx][0];
143787bd33e8SPeter Delevoryas char *name = g_strdup_printf("gpio%s%d", group, pin_idx);
14384b7f9568SRashmica Gupta object_property_add(obj, name, "bool", aspeed_gpio_get_pin,
1439d2623129SMarkus Armbruster aspeed_gpio_set_pin, NULL, NULL);
144015cea92dSPanNengyuan g_free(name);
14414b7f9568SRashmica Gupta }
14424b7f9568SRashmica Gupta }
144387bd33e8SPeter Delevoryas }
14444b7f9568SRashmica Gupta
14454b7f9568SRashmica Gupta static const VMStateDescription vmstate_gpio_regs = {
14464b7f9568SRashmica Gupta .name = TYPE_ASPEED_GPIO"/regs",
14474b7f9568SRashmica Gupta .version_id = 1,
14484b7f9568SRashmica Gupta .minimum_version_id = 1,
14493b9e779bSRichard Henderson .fields = (const VMStateField[]) {
14504b7f9568SRashmica Gupta VMSTATE_UINT32(data_value, GPIOSets),
14514b7f9568SRashmica Gupta VMSTATE_UINT32(data_read, GPIOSets),
14524b7f9568SRashmica Gupta VMSTATE_UINT32(direction, GPIOSets),
14534b7f9568SRashmica Gupta VMSTATE_UINT32(int_enable, GPIOSets),
14544b7f9568SRashmica Gupta VMSTATE_UINT32(int_sens_0, GPIOSets),
14554b7f9568SRashmica Gupta VMSTATE_UINT32(int_sens_1, GPIOSets),
14564b7f9568SRashmica Gupta VMSTATE_UINT32(int_sens_2, GPIOSets),
14574b7f9568SRashmica Gupta VMSTATE_UINT32(int_status, GPIOSets),
14584b7f9568SRashmica Gupta VMSTATE_UINT32(reset_tol, GPIOSets),
14594b7f9568SRashmica Gupta VMSTATE_UINT32(cmd_source_0, GPIOSets),
14604b7f9568SRashmica Gupta VMSTATE_UINT32(cmd_source_1, GPIOSets),
14614b7f9568SRashmica Gupta VMSTATE_UINT32(debounce_1, GPIOSets),
14624b7f9568SRashmica Gupta VMSTATE_UINT32(debounce_2, GPIOSets),
14634b7f9568SRashmica Gupta VMSTATE_UINT32(input_mask, GPIOSets),
14644b7f9568SRashmica Gupta VMSTATE_END_OF_LIST(),
14654b7f9568SRashmica Gupta }
14664b7f9568SRashmica Gupta };
14674b7f9568SRashmica Gupta
14684b7f9568SRashmica Gupta static const VMStateDescription vmstate_aspeed_gpio = {
14694b7f9568SRashmica Gupta .name = TYPE_ASPEED_GPIO,
14704b7f9568SRashmica Gupta .version_id = 1,
14714b7f9568SRashmica Gupta .minimum_version_id = 1,
14723b9e779bSRichard Henderson .fields = (const VMStateField[]) {
14734b7f9568SRashmica Gupta VMSTATE_STRUCT_ARRAY(sets, AspeedGPIOState, ASPEED_GPIO_MAX_NR_SETS,
14744b7f9568SRashmica Gupta 1, vmstate_gpio_regs, GPIOSets),
14754b7f9568SRashmica Gupta VMSTATE_UINT32_ARRAY(debounce_regs, AspeedGPIOState,
14764b7f9568SRashmica Gupta ASPEED_GPIO_NR_DEBOUNCE_REGS),
14774b7f9568SRashmica Gupta VMSTATE_END_OF_LIST(),
14784b7f9568SRashmica Gupta }
14794b7f9568SRashmica Gupta };
14804b7f9568SRashmica Gupta
aspeed_gpio_class_init(ObjectClass * klass,void * data)14814b7f9568SRashmica Gupta static void aspeed_gpio_class_init(ObjectClass *klass, void *data)
14824b7f9568SRashmica Gupta {
14834b7f9568SRashmica Gupta DeviceClass *dc = DEVICE_CLASS(klass);
14844b7f9568SRashmica Gupta
14854b7f9568SRashmica Gupta dc->realize = aspeed_gpio_realize;
1486e3d08143SPeter Maydell device_class_set_legacy_reset(dc, aspeed_gpio_reset);
14874b7f9568SRashmica Gupta dc->desc = "Aspeed GPIO Controller";
14884b7f9568SRashmica Gupta dc->vmsd = &vmstate_aspeed_gpio;
14894b7f9568SRashmica Gupta }
14904b7f9568SRashmica Gupta
aspeed_gpio_ast2400_class_init(ObjectClass * klass,void * data)14914b7f9568SRashmica Gupta static void aspeed_gpio_ast2400_class_init(ObjectClass *klass, void *data)
14924b7f9568SRashmica Gupta {
14934b7f9568SRashmica Gupta AspeedGPIOClass *agc = ASPEED_GPIO_CLASS(klass);
14944b7f9568SRashmica Gupta
14954b7f9568SRashmica Gupta agc->props = ast2400_set_props;
14964b7f9568SRashmica Gupta agc->nr_gpio_pins = 216;
14974b7f9568SRashmica Gupta agc->nr_gpio_sets = 7;
149898edb134SJoel Stanley agc->reg_table = aspeed_3_3v_gpios;
149987511bb8SZheyu Ma agc->reg_table_count = GPIO_3_3V_REG_ARRAY_SIZE;
15009422dbd1SJamin Lin agc->mem_size = 0x1000;
1501404e7534SJamin Lin agc->reg_ops = &aspeed_gpio_ops;
15024b7f9568SRashmica Gupta }
15034b7f9568SRashmica Gupta
aspeed_gpio_2500_class_init(ObjectClass * klass,void * data)15044b7f9568SRashmica Gupta static void aspeed_gpio_2500_class_init(ObjectClass *klass, void *data)
15054b7f9568SRashmica Gupta {
15064b7f9568SRashmica Gupta AspeedGPIOClass *agc = ASPEED_GPIO_CLASS(klass);
15074b7f9568SRashmica Gupta
15084b7f9568SRashmica Gupta agc->props = ast2500_set_props;
15094b7f9568SRashmica Gupta agc->nr_gpio_pins = 228;
15104b7f9568SRashmica Gupta agc->nr_gpio_sets = 8;
151198edb134SJoel Stanley agc->reg_table = aspeed_3_3v_gpios;
151287511bb8SZheyu Ma agc->reg_table_count = GPIO_3_3V_REG_ARRAY_SIZE;
15139422dbd1SJamin Lin agc->mem_size = 0x1000;
1514404e7534SJamin Lin agc->reg_ops = &aspeed_gpio_ops;
15154b7f9568SRashmica Gupta }
15164b7f9568SRashmica Gupta
aspeed_gpio_ast2600_3_3v_class_init(ObjectClass * klass,void * data)151798edb134SJoel Stanley static void aspeed_gpio_ast2600_3_3v_class_init(ObjectClass *klass, void *data)
151836d737eeSRashmica Gupta {
151936d737eeSRashmica Gupta AspeedGPIOClass *agc = ASPEED_GPIO_CLASS(klass);
152036d737eeSRashmica Gupta
152198edb134SJoel Stanley agc->props = ast2600_3_3v_set_props;
152236d737eeSRashmica Gupta agc->nr_gpio_pins = 208;
152336d737eeSRashmica Gupta agc->nr_gpio_sets = 7;
152498edb134SJoel Stanley agc->reg_table = aspeed_3_3v_gpios;
152587511bb8SZheyu Ma agc->reg_table_count = GPIO_3_3V_REG_ARRAY_SIZE;
15269422dbd1SJamin Lin agc->mem_size = 0x800;
1527404e7534SJamin Lin agc->reg_ops = &aspeed_gpio_ops;
152836d737eeSRashmica Gupta }
152936d737eeSRashmica Gupta
aspeed_gpio_ast2600_1_8v_class_init(ObjectClass * klass,void * data)153036d737eeSRashmica Gupta static void aspeed_gpio_ast2600_1_8v_class_init(ObjectClass *klass, void *data)
153136d737eeSRashmica Gupta {
153236d737eeSRashmica Gupta AspeedGPIOClass *agc = ASPEED_GPIO_CLASS(klass);
153336d737eeSRashmica Gupta
153436d737eeSRashmica Gupta agc->props = ast2600_1_8v_set_props;
153536d737eeSRashmica Gupta agc->nr_gpio_pins = 36;
153636d737eeSRashmica Gupta agc->nr_gpio_sets = 2;
153736d737eeSRashmica Gupta agc->reg_table = aspeed_1_8v_gpios;
153887511bb8SZheyu Ma agc->reg_table_count = GPIO_1_8V_REG_ARRAY_SIZE;
15399422dbd1SJamin Lin agc->mem_size = 0x800;
1540404e7534SJamin Lin agc->reg_ops = &aspeed_gpio_ops;
154136d737eeSRashmica Gupta }
154236d737eeSRashmica Gupta
aspeed_gpio_1030_class_init(ObjectClass * klass,void * data)154317075ef2SJamin Lin static void aspeed_gpio_1030_class_init(ObjectClass *klass, void *data)
154417075ef2SJamin Lin {
154517075ef2SJamin Lin AspeedGPIOClass *agc = ASPEED_GPIO_CLASS(klass);
154617075ef2SJamin Lin
154717075ef2SJamin Lin agc->props = ast1030_set_props;
154817075ef2SJamin Lin agc->nr_gpio_pins = 151;
154917075ef2SJamin Lin agc->nr_gpio_sets = 6;
155017075ef2SJamin Lin agc->reg_table = aspeed_3_3v_gpios;
155187511bb8SZheyu Ma agc->reg_table_count = GPIO_3_3V_REG_ARRAY_SIZE;
15529422dbd1SJamin Lin agc->mem_size = 0x1000;
1553404e7534SJamin Lin agc->reg_ops = &aspeed_gpio_ops;
155417075ef2SJamin Lin }
155517075ef2SJamin Lin
aspeed_gpio_2700_class_init(ObjectClass * klass,void * data)1556bac69883SJamin Lin static void aspeed_gpio_2700_class_init(ObjectClass *klass, void *data)
1557bac69883SJamin Lin {
1558bac69883SJamin Lin AspeedGPIOClass *agc = ASPEED_GPIO_CLASS(klass);
1559bac69883SJamin Lin
1560bac69883SJamin Lin agc->props = ast2700_set_props;
1561bac69883SJamin Lin agc->nr_gpio_pins = 216;
1562bac69883SJamin Lin agc->nr_gpio_sets = 7;
1563bac69883SJamin Lin agc->reg_table_count = GPIO_2700_REG_ARRAY_SIZE;
1564bac69883SJamin Lin agc->mem_size = 0x1000;
1565bac69883SJamin Lin agc->reg_ops = &aspeed_gpio_2700_ops;
1566bac69883SJamin Lin }
1567bac69883SJamin Lin
15684b7f9568SRashmica Gupta static const TypeInfo aspeed_gpio_info = {
15694b7f9568SRashmica Gupta .name = TYPE_ASPEED_GPIO,
15704b7f9568SRashmica Gupta .parent = TYPE_SYS_BUS_DEVICE,
15714b7f9568SRashmica Gupta .instance_size = sizeof(AspeedGPIOState),
15724b7f9568SRashmica Gupta .class_size = sizeof(AspeedGPIOClass),
15734b7f9568SRashmica Gupta .class_init = aspeed_gpio_class_init,
15744b7f9568SRashmica Gupta .abstract = true,
15754b7f9568SRashmica Gupta };
15764b7f9568SRashmica Gupta
15774b7f9568SRashmica Gupta static const TypeInfo aspeed_gpio_ast2400_info = {
15784b7f9568SRashmica Gupta .name = TYPE_ASPEED_GPIO "-ast2400",
15794b7f9568SRashmica Gupta .parent = TYPE_ASPEED_GPIO,
15804b7f9568SRashmica Gupta .class_init = aspeed_gpio_ast2400_class_init,
15814b7f9568SRashmica Gupta .instance_init = aspeed_gpio_init,
15824b7f9568SRashmica Gupta };
15834b7f9568SRashmica Gupta
15844b7f9568SRashmica Gupta static const TypeInfo aspeed_gpio_ast2500_info = {
15854b7f9568SRashmica Gupta .name = TYPE_ASPEED_GPIO "-ast2500",
15864b7f9568SRashmica Gupta .parent = TYPE_ASPEED_GPIO,
15874b7f9568SRashmica Gupta .class_init = aspeed_gpio_2500_class_init,
15884b7f9568SRashmica Gupta .instance_init = aspeed_gpio_init,
15894b7f9568SRashmica Gupta };
15904b7f9568SRashmica Gupta
159198edb134SJoel Stanley static const TypeInfo aspeed_gpio_ast2600_3_3v_info = {
159236d737eeSRashmica Gupta .name = TYPE_ASPEED_GPIO "-ast2600",
159336d737eeSRashmica Gupta .parent = TYPE_ASPEED_GPIO,
159498edb134SJoel Stanley .class_init = aspeed_gpio_ast2600_3_3v_class_init,
159536d737eeSRashmica Gupta .instance_init = aspeed_gpio_init,
159636d737eeSRashmica Gupta };
159736d737eeSRashmica Gupta
159836d737eeSRashmica Gupta static const TypeInfo aspeed_gpio_ast2600_1_8v_info = {
159936d737eeSRashmica Gupta .name = TYPE_ASPEED_GPIO "-ast2600-1_8v",
160036d737eeSRashmica Gupta .parent = TYPE_ASPEED_GPIO,
160136d737eeSRashmica Gupta .class_init = aspeed_gpio_ast2600_1_8v_class_init,
160236d737eeSRashmica Gupta .instance_init = aspeed_gpio_init,
160336d737eeSRashmica Gupta };
160436d737eeSRashmica Gupta
160517075ef2SJamin Lin static const TypeInfo aspeed_gpio_ast1030_info = {
160617075ef2SJamin Lin .name = TYPE_ASPEED_GPIO "-ast1030",
160717075ef2SJamin Lin .parent = TYPE_ASPEED_GPIO,
160817075ef2SJamin Lin .class_init = aspeed_gpio_1030_class_init,
160917075ef2SJamin Lin .instance_init = aspeed_gpio_init,
161017075ef2SJamin Lin };
161117075ef2SJamin Lin
1612bac69883SJamin Lin static const TypeInfo aspeed_gpio_ast2700_info = {
1613bac69883SJamin Lin .name = TYPE_ASPEED_GPIO "-ast2700",
1614bac69883SJamin Lin .parent = TYPE_ASPEED_GPIO,
1615bac69883SJamin Lin .class_init = aspeed_gpio_2700_class_init,
1616bac69883SJamin Lin .instance_init = aspeed_gpio_init,
1617bac69883SJamin Lin };
1618bac69883SJamin Lin
aspeed_gpio_register_types(void)16194b7f9568SRashmica Gupta static void aspeed_gpio_register_types(void)
16204b7f9568SRashmica Gupta {
16214b7f9568SRashmica Gupta type_register_static(&aspeed_gpio_info);
16224b7f9568SRashmica Gupta type_register_static(&aspeed_gpio_ast2400_info);
16234b7f9568SRashmica Gupta type_register_static(&aspeed_gpio_ast2500_info);
162498edb134SJoel Stanley type_register_static(&aspeed_gpio_ast2600_3_3v_info);
162536d737eeSRashmica Gupta type_register_static(&aspeed_gpio_ast2600_1_8v_info);
162617075ef2SJamin Lin type_register_static(&aspeed_gpio_ast1030_info);
1627bac69883SJamin Lin type_register_static(&aspeed_gpio_ast2700_info);
16284b7f9568SRashmica Gupta }
16294b7f9568SRashmica Gupta
16304b7f9568SRashmica Gupta type_init(aspeed_gpio_register_types);
1631