Merge remote-tracking branch 'remotes/dgilbert-gitlab/tags/pull-migration-20210726a...
[qemu/armbru.git] / tests / qtest / npcm7xx_gpio-test.c
blob1004cef8121efad7098d2c7be17efc502f5b3a04
1 /*
2 * QTest testcase for the Nuvoton NPCM7xx GPIO modules.
4 * Copyright 2020 Google LLC
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.
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.
17 #include "qemu/osdep.h"
18 #include "libqtest-single.h"
20 #define NR_GPIO_DEVICES (8)
21 #define GPIO(x) (0xf0010000 + (x) * 0x1000)
22 #define GPIO_IRQ(x) (116 + (x))
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
56 static void gpio_unlock(int n)
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);
64 /* Restore the GPIO controller to a sensible default state. */
65 static void gpio_reset(int n)
67 gpio_unlock(0);
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);
80 static void test_dout_to_din(void)
82 gpio_reset(0);
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);
94 static void test_pullup_pulldown(void)
96 gpio_reset(0);
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.
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);
113 static void test_output_enable(void)
115 gpio_reset(0);
118 * With all pins weakly pulled down, and DOUT all-ones, OE should be
119 * reflected on DIN.
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);
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);
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);
137 static void test_open_drain(void)
139 gpio_reset(0);
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.
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);
156 static void test_polarity(void)
158 gpio_reset(0);
161 * In push-pull mode, DIN should reflect DOUT because the signal is
162 * inverted in both directions.
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);
172 * When turning off the drivers, DIN should reflect the inverse of the
173 * pulled-up lines.
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);
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.
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);
193 static void test_input_mask(void)
195 gpio_reset(0);
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);
205 static void test_temp_lock(void)
207 gpio_reset(0);
209 writel(GPIO(0) + GP_N_DOUT, 0x98765432);
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);
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);
236 static void test_events_level(void)
238 gpio_reset(0);
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);
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)));
258 static void test_events_rising_edge(void)
260 gpio_reset(0);
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);
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)));
284 static void test_events_both_edges(void)
286 gpio_reset(0);
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);
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)));
310 static void test_gpion_irq(gconstpointer test_data)
312 intptr_t n = (intptr_t)test_data;
314 gpio_reset(n);
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);
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)));
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)));
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)));
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)));
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)));
354 int main(int argc, char **argv)
356 int ret;
357 int i;
359 g_test_init(&argc, &argv, NULL);
360 g_test_set_nonfatal_assertions();
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);
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);
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();
384 return ret;