xref: /qemu/tests/qtest/npcm7xx_gpio-test.c (revision 802427bcdae1ad2eceea8a8877ecad835e3f8fde)
1*526dbbe0SHavard Skinnemoen /*
2*526dbbe0SHavard Skinnemoen  * QTest testcase for the Nuvoton NPCM7xx GPIO modules.
3*526dbbe0SHavard Skinnemoen  *
4*526dbbe0SHavard Skinnemoen  * Copyright 2020 Google LLC
5*526dbbe0SHavard Skinnemoen  *
6*526dbbe0SHavard Skinnemoen  * This program is free software; you can redistribute it and/or modify it
7*526dbbe0SHavard Skinnemoen  * under the terms of the GNU General Public License as published by the
8*526dbbe0SHavard Skinnemoen  * Free Software Foundation; either version 2 of the License, or
9*526dbbe0SHavard Skinnemoen  * (at your option) any later version.
10*526dbbe0SHavard Skinnemoen  *
11*526dbbe0SHavard Skinnemoen  * This program is distributed in the hope that it will be useful, but WITHOUT
12*526dbbe0SHavard Skinnemoen  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13*526dbbe0SHavard Skinnemoen  * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14*526dbbe0SHavard Skinnemoen  * for more details.
15*526dbbe0SHavard Skinnemoen  */
16*526dbbe0SHavard Skinnemoen 
17*526dbbe0SHavard Skinnemoen #include "qemu/osdep.h"
18*526dbbe0SHavard Skinnemoen #include "libqtest-single.h"
19*526dbbe0SHavard Skinnemoen 
20*526dbbe0SHavard Skinnemoen #define NR_GPIO_DEVICES (8)
21*526dbbe0SHavard Skinnemoen #define GPIO(x)         (0xf0010000 + (x) * 0x1000)
22*526dbbe0SHavard Skinnemoen #define GPIO_IRQ(x)     (116 + (x))
23*526dbbe0SHavard Skinnemoen 
24*526dbbe0SHavard Skinnemoen /* GPIO registers */
25*526dbbe0SHavard Skinnemoen #define GP_N_TLOCK1     0x00
26*526dbbe0SHavard Skinnemoen #define GP_N_DIN        0x04 /* Data IN */
27*526dbbe0SHavard Skinnemoen #define GP_N_POL        0x08 /* Polarity */
28*526dbbe0SHavard Skinnemoen #define GP_N_DOUT       0x0c /* Data OUT */
29*526dbbe0SHavard Skinnemoen #define GP_N_OE         0x10 /* Output Enable */
30*526dbbe0SHavard Skinnemoen #define GP_N_OTYP       0x14
31*526dbbe0SHavard Skinnemoen #define GP_N_MP         0x18
32*526dbbe0SHavard Skinnemoen #define GP_N_PU         0x1c /* Pull-up */
33*526dbbe0SHavard Skinnemoen #define GP_N_PD         0x20 /* Pull-down */
34*526dbbe0SHavard Skinnemoen #define GP_N_DBNC       0x24 /* Debounce */
35*526dbbe0SHavard Skinnemoen #define GP_N_EVTYP      0x28 /* Event Type */
36*526dbbe0SHavard Skinnemoen #define GP_N_EVBE       0x2c /* Event Both Edge */
37*526dbbe0SHavard Skinnemoen #define GP_N_OBL0       0x30
38*526dbbe0SHavard Skinnemoen #define GP_N_OBL1       0x34
39*526dbbe0SHavard Skinnemoen #define GP_N_OBL2       0x38
40*526dbbe0SHavard Skinnemoen #define GP_N_OBL3       0x3c
41*526dbbe0SHavard Skinnemoen #define GP_N_EVEN       0x40 /* Event Enable */
42*526dbbe0SHavard Skinnemoen #define GP_N_EVENS      0x44 /* Event Set (enable) */
43*526dbbe0SHavard Skinnemoen #define GP_N_EVENC      0x48 /* Event Clear (disable) */
44*526dbbe0SHavard Skinnemoen #define GP_N_EVST       0x4c /* Event Status */
45*526dbbe0SHavard Skinnemoen #define GP_N_SPLCK      0x50
46*526dbbe0SHavard Skinnemoen #define GP_N_MPLCK      0x54
47*526dbbe0SHavard Skinnemoen #define GP_N_IEM        0x58 /* Input Enable */
48*526dbbe0SHavard Skinnemoen #define GP_N_OSRC       0x5c
49*526dbbe0SHavard Skinnemoen #define GP_N_ODSC       0x60
50*526dbbe0SHavard Skinnemoen #define GP_N_DOS        0x68 /* Data OUT Set */
51*526dbbe0SHavard Skinnemoen #define GP_N_DOC        0x6c /* Data OUT Clear */
52*526dbbe0SHavard Skinnemoen #define GP_N_OES        0x70 /* Output Enable Set */
53*526dbbe0SHavard Skinnemoen #define GP_N_OEC        0x74 /* Output Enable Clear */
54*526dbbe0SHavard Skinnemoen #define GP_N_TLOCK2     0x7c
55*526dbbe0SHavard Skinnemoen 
gpio_unlock(int n)56*526dbbe0SHavard Skinnemoen static void gpio_unlock(int n)
57*526dbbe0SHavard Skinnemoen {
58*526dbbe0SHavard Skinnemoen     if (readl(GPIO(n) + GP_N_TLOCK1) != 0) {
59*526dbbe0SHavard Skinnemoen         writel(GPIO(n) + GP_N_TLOCK2, 0xc0de1248);
60*526dbbe0SHavard Skinnemoen         writel(GPIO(n) + GP_N_TLOCK1, 0xc0defa73);
61*526dbbe0SHavard Skinnemoen     }
62*526dbbe0SHavard Skinnemoen }
63*526dbbe0SHavard Skinnemoen 
64*526dbbe0SHavard Skinnemoen /* Restore the GPIO controller to a sensible default state. */
gpio_reset(int n)65*526dbbe0SHavard Skinnemoen static void gpio_reset(int n)
66*526dbbe0SHavard Skinnemoen {
67*526dbbe0SHavard Skinnemoen     gpio_unlock(0);
68*526dbbe0SHavard Skinnemoen 
69*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_EVEN, 0x00000000);
70*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_EVST, 0xffffffff);
71*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_POL, 0x00000000);
72*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_DOUT, 0x00000000);
73*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_OE, 0x00000000);
74*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_OTYP, 0x00000000);
75*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_PU, 0xffffffff);
76*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_PD, 0x00000000);
77*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_IEM, 0xffffffff);
78*526dbbe0SHavard Skinnemoen }
79*526dbbe0SHavard Skinnemoen 
test_dout_to_din(void)80*526dbbe0SHavard Skinnemoen static void test_dout_to_din(void)
81*526dbbe0SHavard Skinnemoen {
82*526dbbe0SHavard Skinnemoen     gpio_reset(0);
83*526dbbe0SHavard Skinnemoen 
84*526dbbe0SHavard Skinnemoen     /* When output is enabled, DOUT should be reflected on DIN. */
85*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OE, 0xffffffff);
86*526dbbe0SHavard Skinnemoen     /* PU and PD shouldn't have any impact on DIN. */
87*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PU, 0xffff0000);
88*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PD, 0x0000ffff);
89*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0x12345678);
90*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DOUT), ==, 0x12345678);
91*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DIN), ==, 0x12345678);
92*526dbbe0SHavard Skinnemoen }
93*526dbbe0SHavard Skinnemoen 
test_pullup_pulldown(void)94*526dbbe0SHavard Skinnemoen static void test_pullup_pulldown(void)
95*526dbbe0SHavard Skinnemoen {
96*526dbbe0SHavard Skinnemoen     gpio_reset(0);
97*526dbbe0SHavard Skinnemoen 
98*526dbbe0SHavard Skinnemoen     /*
99*526dbbe0SHavard Skinnemoen      * When output is disabled, and PD is the inverse of PU, PU should be
100*526dbbe0SHavard Skinnemoen      * reflected on DIN. If PD is not the inverse of PU, the state of DIN is
101*526dbbe0SHavard Skinnemoen      * undefined, so we don't test that.
102*526dbbe0SHavard Skinnemoen      */
103*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OE, 0x00000000);
104*526dbbe0SHavard Skinnemoen     /* DOUT shouldn't have any impact on DIN. */
105*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xffff0000);
106*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PU, 0x23456789);
107*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PD, ~0x23456789U);
108*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_PU), ==, 0x23456789);
109*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_PD), ==, ~0x23456789U);
110*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DIN), ==, 0x23456789);
111*526dbbe0SHavard Skinnemoen }
112*526dbbe0SHavard Skinnemoen 
test_output_enable(void)113*526dbbe0SHavard Skinnemoen static void test_output_enable(void)
114*526dbbe0SHavard Skinnemoen {
115*526dbbe0SHavard Skinnemoen     gpio_reset(0);
116*526dbbe0SHavard Skinnemoen 
117*526dbbe0SHavard Skinnemoen     /*
118*526dbbe0SHavard Skinnemoen      * With all pins weakly pulled down, and DOUT all-ones, OE should be
119*526dbbe0SHavard Skinnemoen      * reflected on DIN.
120*526dbbe0SHavard Skinnemoen      */
121*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xffffffff);
122*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PU, 0x00000000);
123*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PD, 0xffffffff);
124*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OE, 0x3456789a);
125*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_OE), ==, 0x3456789a);
126*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DIN), ==, 0x3456789a);
127*526dbbe0SHavard Skinnemoen 
128*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OEC, 0x00030002);
129*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_OE), ==, 0x34547898);
130*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DIN), ==, 0x34547898);
131*526dbbe0SHavard Skinnemoen 
132*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OES, 0x0000f001);
133*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_OE), ==, 0x3454f899);
134*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DIN), ==, 0x3454f899);
135*526dbbe0SHavard Skinnemoen }
136*526dbbe0SHavard Skinnemoen 
test_open_drain(void)137*526dbbe0SHavard Skinnemoen static void test_open_drain(void)
138*526dbbe0SHavard Skinnemoen {
139*526dbbe0SHavard Skinnemoen     gpio_reset(0);
140*526dbbe0SHavard Skinnemoen 
141*526dbbe0SHavard Skinnemoen     /*
142*526dbbe0SHavard Skinnemoen      * Upper half of DOUT drives a 1 only if the corresponding bit in OTYP is
143*526dbbe0SHavard Skinnemoen      * not set. If OTYP is set, DIN is determined by PU/PD. Lower half of
144*526dbbe0SHavard Skinnemoen      * DOUT always drives a 0 regardless of OTYP; PU/PD have no effect.  When
145*526dbbe0SHavard Skinnemoen      * OE is 0, output is determined by PU/PD; OTYP has no effect.
146*526dbbe0SHavard Skinnemoen      */
147*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OTYP, 0x456789ab);
148*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OE, 0xf0f0f0f0);
149*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xffff0000);
150*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PU, 0xff00ff00);
151*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PD, 0x00ff00ff);
152*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_OTYP), ==, 0x456789ab);
153*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DIN), ==, 0xff900f00);
154*526dbbe0SHavard Skinnemoen }
155*526dbbe0SHavard Skinnemoen 
test_polarity(void)156*526dbbe0SHavard Skinnemoen static void test_polarity(void)
157*526dbbe0SHavard Skinnemoen {
158*526dbbe0SHavard Skinnemoen     gpio_reset(0);
159*526dbbe0SHavard Skinnemoen 
160*526dbbe0SHavard Skinnemoen     /*
161*526dbbe0SHavard Skinnemoen      * In push-pull mode, DIN should reflect DOUT because the signal is
162*526dbbe0SHavard Skinnemoen      * inverted in both directions.
163*526dbbe0SHavard Skinnemoen      */
164*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OTYP, 0x00000000);
165*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OE, 0xffffffff);
166*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0x56789abc);
167*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_POL, 0x6789abcd);
168*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_POL), ==, 0x6789abcd);
169*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DIN), ==, 0x56789abc);
170*526dbbe0SHavard Skinnemoen 
171*526dbbe0SHavard Skinnemoen     /*
172*526dbbe0SHavard Skinnemoen      * When turning off the drivers, DIN should reflect the inverse of the
173*526dbbe0SHavard Skinnemoen      * pulled-up lines.
174*526dbbe0SHavard Skinnemoen      */
175*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OE, 0x00000000);
176*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_POL, 0xffffffff);
177*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PU, 0x789abcde);
178*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PD, ~0x789abcdeU);
179*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DIN), ==, ~0x789abcdeU);
180*526dbbe0SHavard Skinnemoen 
181*526dbbe0SHavard Skinnemoen     /*
182*526dbbe0SHavard Skinnemoen      * In open-drain mode, DOUT=1 will appear to drive the pin high (since DIN
183*526dbbe0SHavard Skinnemoen      * is inverted), while DOUT=0 will leave the pin floating.
184*526dbbe0SHavard Skinnemoen      */
185*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OTYP, 0xffffffff);
186*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OE, 0xffffffff);
187*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PU, 0xffff0000);
188*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_PD, 0x0000ffff);
189*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xff00ff00);
190*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DIN), ==, 0xff00ffff);
191*526dbbe0SHavard Skinnemoen }
192*526dbbe0SHavard Skinnemoen 
test_input_mask(void)193*526dbbe0SHavard Skinnemoen static void test_input_mask(void)
194*526dbbe0SHavard Skinnemoen {
195*526dbbe0SHavard Skinnemoen     gpio_reset(0);
196*526dbbe0SHavard Skinnemoen 
197*526dbbe0SHavard Skinnemoen     /* IEM=0 forces the input to zero before polarity inversion. */
198*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OE, 0xffffffff);
199*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xff00ff00);
200*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_POL, 0xffff0000);
201*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_IEM, 0x87654321);
202*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DIN), ==, 0xff9a4300);
203*526dbbe0SHavard Skinnemoen }
204*526dbbe0SHavard Skinnemoen 
test_temp_lock(void)205*526dbbe0SHavard Skinnemoen static void test_temp_lock(void)
206*526dbbe0SHavard Skinnemoen {
207*526dbbe0SHavard Skinnemoen     gpio_reset(0);
208*526dbbe0SHavard Skinnemoen 
209*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0x98765432);
210*526dbbe0SHavard Skinnemoen 
211*526dbbe0SHavard Skinnemoen     /* Make sure we're unlocked initially. */
212*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_TLOCK1), ==, 0);
213*526dbbe0SHavard Skinnemoen     /* Writing any value to TLOCK1 will lock. */
214*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_TLOCK1, 0);
215*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_TLOCK1), ==, 1);
216*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xa9876543);
217*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DOUT), ==, 0x98765432);
218*526dbbe0SHavard Skinnemoen     /* Now, try to unlock. */
219*526dbbe0SHavard Skinnemoen     gpio_unlock(0);
220*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_TLOCK1), ==, 0);
221*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xa9876543);
222*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DOUT), ==, 0xa9876543);
223*526dbbe0SHavard Skinnemoen 
224*526dbbe0SHavard Skinnemoen     /* Try it again, but write TLOCK2 to lock. */
225*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_TLOCK2, 0);
226*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_TLOCK1), ==, 1);
227*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0x98765432);
228*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DOUT), ==, 0xa9876543);
229*526dbbe0SHavard Skinnemoen     /* Now, try to unlock. */
230*526dbbe0SHavard Skinnemoen     gpio_unlock(0);
231*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_TLOCK1), ==, 0);
232*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0x98765432);
233*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_DOUT), ==, 0x98765432);
234*526dbbe0SHavard Skinnemoen }
235*526dbbe0SHavard Skinnemoen 
test_events_level(void)236*526dbbe0SHavard Skinnemoen static void test_events_level(void)
237*526dbbe0SHavard Skinnemoen {
238*526dbbe0SHavard Skinnemoen     gpio_reset(0);
239*526dbbe0SHavard Skinnemoen 
240*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVTYP, 0x00000000);
241*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xba987654);
242*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OE, 0xffffffff);
243*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVST, 0xffffffff);
244*526dbbe0SHavard Skinnemoen 
245*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0xba987654);
246*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
247*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0x00000000);
248*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0xba987654);
249*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
250*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVST, 0x00007654);
251*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0xba980000);
252*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
253*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVST, 0xba980000);
254*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0x00000000);
255*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
256*526dbbe0SHavard Skinnemoen }
257*526dbbe0SHavard Skinnemoen 
test_events_rising_edge(void)258*526dbbe0SHavard Skinnemoen static void test_events_rising_edge(void)
259*526dbbe0SHavard Skinnemoen {
260*526dbbe0SHavard Skinnemoen     gpio_reset(0);
261*526dbbe0SHavard Skinnemoen 
262*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVTYP, 0xffffffff);
263*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVBE, 0x00000000);
264*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xffff0000);
265*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OE, 0xffffffff);
266*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVST, 0xffffffff);
267*526dbbe0SHavard Skinnemoen 
268*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0x00000000);
269*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
270*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xff00ff00);
271*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0x0000ff00);
272*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
273*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0x00ff0000);
274*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0x00ffff00);
275*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
276*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVST, 0x0000f000);
277*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0x00ff0f00);
278*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
279*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVST, 0x00ff0f00);
280*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0x00000000);
281*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
282*526dbbe0SHavard Skinnemoen }
283*526dbbe0SHavard Skinnemoen 
test_events_both_edges(void)284*526dbbe0SHavard Skinnemoen static void test_events_both_edges(void)
285*526dbbe0SHavard Skinnemoen {
286*526dbbe0SHavard Skinnemoen     gpio_reset(0);
287*526dbbe0SHavard Skinnemoen 
288*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVTYP, 0xffffffff);
289*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVBE, 0xffffffff);
290*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xffff0000);
291*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_OE, 0xffffffff);
292*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVST, 0xffffffff);
293*526dbbe0SHavard Skinnemoen 
294*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0x00000000);
295*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
296*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xff00ff00);
297*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0x00ffff00);
298*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
299*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_DOUT, 0xef00ff08);
300*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0x10ffff08);
301*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
302*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVST, 0x0000f000);
303*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0x10ff0f08);
304*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
305*526dbbe0SHavard Skinnemoen     writel(GPIO(0) + GP_N_EVST, 0x10ff0f08);
306*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(0) + GP_N_EVST), ==, 0x00000000);
307*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(0)));
308*526dbbe0SHavard Skinnemoen }
309*526dbbe0SHavard Skinnemoen 
test_gpion_irq(gconstpointer test_data)310*526dbbe0SHavard Skinnemoen static void test_gpion_irq(gconstpointer test_data)
311*526dbbe0SHavard Skinnemoen {
312*526dbbe0SHavard Skinnemoen     intptr_t n = (intptr_t)test_data;
313*526dbbe0SHavard Skinnemoen 
314*526dbbe0SHavard Skinnemoen     gpio_reset(n);
315*526dbbe0SHavard Skinnemoen 
316*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_EVTYP, 0x00000000);
317*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_DOUT, 0x00000000);
318*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_OE, 0xffffffff);
319*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_EVST, 0xffffffff);
320*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_EVEN, 0x00000000);
321*526dbbe0SHavard Skinnemoen 
322*526dbbe0SHavard Skinnemoen     /* Trigger an event; interrupts are masked. */
323*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(n) + GP_N_EVST), ==, 0x00000000);
324*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(n)));
325*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_DOS, 0x00008000);
326*526dbbe0SHavard Skinnemoen     g_assert_cmphex(readl(GPIO(n) + GP_N_EVST), ==, 0x00008000);
327*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(n)));
328*526dbbe0SHavard Skinnemoen 
329*526dbbe0SHavard Skinnemoen     /* Unmask all event interrupts; verify that the interrupt fired. */
330*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_EVEN, 0xffffffff);
331*526dbbe0SHavard Skinnemoen     g_assert_true(qtest_get_irq(global_qtest, GPIO_IRQ(n)));
332*526dbbe0SHavard Skinnemoen 
333*526dbbe0SHavard Skinnemoen     /* Clear the current bit, set a new bit, irq stays asserted. */
334*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_DOC, 0x00008000);
335*526dbbe0SHavard Skinnemoen     g_assert_true(qtest_get_irq(global_qtest, GPIO_IRQ(n)));
336*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_DOS, 0x00000200);
337*526dbbe0SHavard Skinnemoen     g_assert_true(qtest_get_irq(global_qtest, GPIO_IRQ(n)));
338*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_EVST, 0x00008000);
339*526dbbe0SHavard Skinnemoen     g_assert_true(qtest_get_irq(global_qtest, GPIO_IRQ(n)));
340*526dbbe0SHavard Skinnemoen 
341*526dbbe0SHavard Skinnemoen     /* Mask/unmask the event that's currently active. */
342*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_EVENC, 0x00000200);
343*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(n)));
344*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_EVENS, 0x00000200);
345*526dbbe0SHavard Skinnemoen     g_assert_true(qtest_get_irq(global_qtest, GPIO_IRQ(n)));
346*526dbbe0SHavard Skinnemoen 
347*526dbbe0SHavard Skinnemoen     /* Clear the input and the status bit, irq is deasserted. */
348*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_DOC, 0x00000200);
349*526dbbe0SHavard Skinnemoen     g_assert_true(qtest_get_irq(global_qtest, GPIO_IRQ(n)));
350*526dbbe0SHavard Skinnemoen     writel(GPIO(n) + GP_N_EVST, 0x00000200);
351*526dbbe0SHavard Skinnemoen     g_assert_false(qtest_get_irq(global_qtest, GPIO_IRQ(n)));
352*526dbbe0SHavard Skinnemoen }
353*526dbbe0SHavard Skinnemoen 
main(int argc,char ** argv)354*526dbbe0SHavard Skinnemoen int main(int argc, char **argv)
355*526dbbe0SHavard Skinnemoen {
356*526dbbe0SHavard Skinnemoen     int ret;
357*526dbbe0SHavard Skinnemoen     int i;
358*526dbbe0SHavard Skinnemoen 
359*526dbbe0SHavard Skinnemoen     g_test_init(&argc, &argv, NULL);
360*526dbbe0SHavard Skinnemoen     g_test_set_nonfatal_assertions();
361*526dbbe0SHavard Skinnemoen 
362*526dbbe0SHavard Skinnemoen     qtest_add_func("/npcm7xx_gpio/dout_to_din", test_dout_to_din);
363*526dbbe0SHavard Skinnemoen     qtest_add_func("/npcm7xx_gpio/pullup_pulldown", test_pullup_pulldown);
364*526dbbe0SHavard Skinnemoen     qtest_add_func("/npcm7xx_gpio/output_enable", test_output_enable);
365*526dbbe0SHavard Skinnemoen     qtest_add_func("/npcm7xx_gpio/open_drain", test_open_drain);
366*526dbbe0SHavard Skinnemoen     qtest_add_func("/npcm7xx_gpio/polarity", test_polarity);
367*526dbbe0SHavard Skinnemoen     qtest_add_func("/npcm7xx_gpio/input_mask", test_input_mask);
368*526dbbe0SHavard Skinnemoen     qtest_add_func("/npcm7xx_gpio/temp_lock", test_temp_lock);
369*526dbbe0SHavard Skinnemoen     qtest_add_func("/npcm7xx_gpio/events/level", test_events_level);
370*526dbbe0SHavard Skinnemoen     qtest_add_func("/npcm7xx_gpio/events/rising_edge", test_events_rising_edge);
371*526dbbe0SHavard Skinnemoen     qtest_add_func("/npcm7xx_gpio/events/both_edges", test_events_both_edges);
372*526dbbe0SHavard Skinnemoen 
373*526dbbe0SHavard Skinnemoen     for (i = 0; i < NR_GPIO_DEVICES; i++) {
374*526dbbe0SHavard Skinnemoen         g_autofree char *test_name =
375*526dbbe0SHavard Skinnemoen             g_strdup_printf("/npcm7xx_gpio/gpio[%d]/irq", i);
376*526dbbe0SHavard Skinnemoen         qtest_add_data_func(test_name, (void *)(intptr_t)i, test_gpion_irq);
377*526dbbe0SHavard Skinnemoen     }
378*526dbbe0SHavard Skinnemoen 
379*526dbbe0SHavard Skinnemoen     qtest_start("-machine npcm750-evb");
380*526dbbe0SHavard Skinnemoen     qtest_irq_intercept_in(global_qtest, "/machine/soc/a9mpcore/gic");
381*526dbbe0SHavard Skinnemoen     ret = g_test_run();
382*526dbbe0SHavard Skinnemoen     qtest_end();
383*526dbbe0SHavard Skinnemoen 
384*526dbbe0SHavard Skinnemoen     return ret;
385*526dbbe0SHavard Skinnemoen }
386