mirror of
				https://github.com/qemu/qemu.git
				synced 2025-10-30 19:15:42 +00:00 
			
		
		
		
	 907b5105f1
			
		
	
	
		907b5105f1
		
	
	
	
	
		
			
			Since commit a2ce7dbd91 ("meson: convert tests/qtest to meson"),
libqtest.h is under libqos/ directory, while libqtest.c is still in
qtest/. Move back to its original location to avoid mixing with libqos/.
Suggested-by: Thomas Huth <thuth@redhat.com>
Signed-off-by: Marc-André Lureau <marcandre.lureau@redhat.com>
Reviewed-by: Thomas Huth <thuth@redhat.com>
Reviewed-by: Stefan Berger <stefanb@linux.ibm.com>
		
	
			
		
			
				
	
	
		
			164 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			164 lines
		
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * QTest testcase for acpi-erst
 | |
|  *
 | |
|  * Copyright (c) 2021 Oracle
 | |
|  *
 | |
|  * This work is licensed under the terms of the GNU GPL, version 2 or later.
 | |
|  * See the COPYING file in the top-level directory.
 | |
|  */
 | |
| 
 | |
| #include "qemu/osdep.h"
 | |
| #include <glib/gstdio.h>
 | |
| #include "libqos/libqos-pc.h"
 | |
| #include "libqtest.h"
 | |
| 
 | |
| #include "hw/pci/pci.h"
 | |
| 
 | |
| static void save_fn(QPCIDevice *dev, int devfn, void *data)
 | |
| {
 | |
|     QPCIDevice **pdev = (QPCIDevice **) data;
 | |
| 
 | |
|     *pdev = dev;
 | |
| }
 | |
| 
 | |
| static QPCIDevice *get_erst_device(QPCIBus *pcibus)
 | |
| {
 | |
|     QPCIDevice *dev;
 | |
| 
 | |
|     dev = NULL;
 | |
|     qpci_device_foreach(pcibus,
 | |
|         PCI_VENDOR_ID_REDHAT,
 | |
|         PCI_DEVICE_ID_REDHAT_ACPI_ERST,
 | |
|         save_fn, &dev);
 | |
|     g_assert(dev != NULL);
 | |
| 
 | |
|     return dev;
 | |
| }
 | |
| 
 | |
| typedef struct _ERSTState {
 | |
|     QOSState *qs;
 | |
|     QPCIBar reg_bar, mem_bar;
 | |
|     uint64_t reg_barsize, mem_barsize;
 | |
|     QPCIDevice *dev;
 | |
| } ERSTState;
 | |
| 
 | |
| #define ACTION 0
 | |
| #define VALUE 8
 | |
| 
 | |
| static const char *reg2str(unsigned reg)
 | |
| {
 | |
|     switch (reg) {
 | |
|     case 0:
 | |
|         return "ACTION";
 | |
|     case 8:
 | |
|         return "VALUE";
 | |
|     default:
 | |
|         return NULL;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static inline uint32_t in_reg32(ERSTState *s, unsigned reg)
 | |
| {
 | |
|     const char *name = reg2str(reg);
 | |
|     uint32_t res;
 | |
| 
 | |
|     res = qpci_io_readl(s->dev, s->reg_bar, reg);
 | |
|     g_test_message("*%s -> %08x", name, res);
 | |
| 
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| static inline uint64_t in_reg64(ERSTState *s, unsigned reg)
 | |
| {
 | |
|     const char *name = reg2str(reg);
 | |
|     uint64_t res;
 | |
| 
 | |
|     res = qpci_io_readq(s->dev, s->reg_bar, reg);
 | |
|     g_test_message("*%s -> %016" PRIx64, name, res);
 | |
| 
 | |
|     return res;
 | |
| }
 | |
| 
 | |
| static inline void out_reg32(ERSTState *s, unsigned reg, uint32_t v)
 | |
| {
 | |
|     const char *name = reg2str(reg);
 | |
| 
 | |
|     g_test_message("%08x -> *%s", v, name);
 | |
|     qpci_io_writel(s->dev, s->reg_bar, reg, v);
 | |
| }
 | |
| 
 | |
| static void cleanup_vm(ERSTState *s)
 | |
| {
 | |
|     g_free(s->dev);
 | |
|     qtest_shutdown(s->qs);
 | |
| }
 | |
| 
 | |
| static void setup_vm_cmd(ERSTState *s, const char *cmd)
 | |
| {
 | |
|     const char *arch = qtest_get_arch();
 | |
| 
 | |
|     if (strcmp(arch, "i386") == 0 || strcmp(arch, "x86_64") == 0) {
 | |
|         s->qs = qtest_pc_boot(cmd);
 | |
|     } else {
 | |
|         g_printerr("erst-test tests are only available on x86\n");
 | |
|         exit(EXIT_FAILURE);
 | |
|     }
 | |
|     s->dev = get_erst_device(s->qs->pcibus);
 | |
| 
 | |
|     s->reg_bar = qpci_iomap(s->dev, 0, &s->reg_barsize);
 | |
|     g_assert_cmpuint(s->reg_barsize, ==, 16);
 | |
| 
 | |
|     s->mem_bar = qpci_iomap(s->dev, 1, &s->mem_barsize);
 | |
|     g_assert_cmpuint(s->mem_barsize, ==, 0x2000);
 | |
| 
 | |
|     qpci_device_enable(s->dev);
 | |
| }
 | |
| 
 | |
| static void test_acpi_erst_basic(void)
 | |
| {
 | |
|     ERSTState state;
 | |
|     uint64_t log_address_range;
 | |
|     uint64_t log_address_length;
 | |
|     uint32_t log_address_attr;
 | |
| 
 | |
|     setup_vm_cmd(&state,
 | |
|         "-object memory-backend-file,"
 | |
|             "mem-path=acpi-erst.XXXXXX,"
 | |
|             "size=64K,"
 | |
|             "share=on,"
 | |
|             "id=nvram "
 | |
|         "-device acpi-erst,"
 | |
|             "memdev=nvram");
 | |
| 
 | |
|     out_reg32(&state, ACTION, 0xD);
 | |
|     log_address_range = in_reg64(&state, VALUE);
 | |
|     out_reg32(&state, ACTION, 0xE);
 | |
|     log_address_length = in_reg64(&state, VALUE);
 | |
|     out_reg32(&state, ACTION, 0xF);
 | |
|     log_address_attr = in_reg32(&state, VALUE);
 | |
| 
 | |
|     /* Check log_address_range is not 0, ~0 or base */
 | |
|     g_assert_cmpuint(log_address_range, !=,  0ULL);
 | |
|     g_assert_cmpuint(log_address_range, !=, ~0ULL);
 | |
|     g_assert_cmpuint(log_address_range, !=, state.reg_bar.addr);
 | |
|     g_assert_cmpuint(log_address_range, ==, state.mem_bar.addr);
 | |
| 
 | |
|     /* Check log_address_length is bar1_size */
 | |
|     g_assert_cmpuint(log_address_length, ==, state.mem_barsize);
 | |
| 
 | |
|     /* Check log_address_attr is 0 */
 | |
|     g_assert_cmpuint(log_address_attr, ==, 0);
 | |
| 
 | |
|     cleanup_vm(&state);
 | |
| }
 | |
| 
 | |
| int main(int argc, char **argv)
 | |
| {
 | |
|     int ret;
 | |
| 
 | |
|     g_test_init(&argc, &argv, NULL);
 | |
|     qtest_add_func("/acpi-erst/basic", test_acpi_erst_basic);
 | |
|     ret = g_test_run();
 | |
|     return ret;
 | |
| }
 |