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