xref: /openbmc/qemu/hw/gpio/aspeed_gpio.c (revision d34fea6cce319c54ff09f96a2e74fb5a2d50affa)
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