qemu/include/hw/gpio/nrf51_gpio.h
Cameron Esfahani c8aeef3aed nrf51: Fix last GPIO CNF address
NRF51_GPIO_REG_CNF_END doesn't actually refer to the start of the last
valid CNF register: it's referring to the last byte of the last valid
CNF register.

This hasn't been a problem up to now, as current implementation in
memory.c turns an unaligned 4-byte read from 0x77f to a single byte read
and the qtest only looks at the least-significant byte of the register.

But when running with patches which fix unaligned accesses in memory.c,
the qtest breaks.

Considering NRF51 doesn't support unaligned accesses, the simplest fix
is to actually set NRF51_GPIO_REG_CNF_END to the start of the last valid
CNF register: 0x77c.

Now, qtests work with or without the unaligned access patches.

Reviewed-by: Cédric Le Goater <clg@kaod.org>
Tested-by: Cédric Le Goater <clg@kaod.org>
Reviewed-by: Joel Stanley <joel@jms.id.au>
Signed-off-by: Cameron Esfahani <dirty@apple.com>
Message-id: 51b427f06838622da783d38ba56e3630d6d85c60.1586925392.git.dirty@apple.com
Reviewed-by: Peter Maydell <peter.maydell@linaro.org>
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
2020-04-30 11:52:27 +01:00

70 lines
1.9 KiB
C

/*
* nRF51 System-on-Chip general purpose input/output register definition
*
* QEMU interface:
* + sysbus MMIO regions 0: GPIO registers
* + Unnamed GPIO inputs 0-31: Set tri-state input level for GPIO pin.
* Level -1: Externally Disconnected/Floating; Pull-up/down will be regarded
* Level 0: Input externally driven LOW
* Level 1: Input externally driven HIGH
* + Unnamed GPIO outputs 0-31:
* Level -1: Disconnected/Floating
* Level 0: Driven LOW
* Level 1: Driven HIGH
*
* Accuracy of the peripheral model:
* + The nRF51 GPIO output driver supports two modes, standard and high-current
* mode. These different drive modes are not modeled and handled the same.
* + Pin SENSEing is not modeled/implemented.
*
* Copyright 2018 Steffen Görtz <contrib@steffen-goertz.de>
*
* This code is licensed under the GPL version 2 or later. See
* the COPYING file in the top-level directory.
*
*/
#ifndef NRF51_GPIO_H
#define NRF51_GPIO_H
#include "hw/sysbus.h"
#define TYPE_NRF51_GPIO "nrf51_soc.gpio"
#define NRF51_GPIO(obj) OBJECT_CHECK(NRF51GPIOState, (obj), TYPE_NRF51_GPIO)
#define NRF51_GPIO_PINS 32
#define NRF51_GPIO_SIZE 0x1000
#define NRF51_GPIO_REG_OUT 0x504
#define NRF51_GPIO_REG_OUTSET 0x508
#define NRF51_GPIO_REG_OUTCLR 0x50C
#define NRF51_GPIO_REG_IN 0x510
#define NRF51_GPIO_REG_DIR 0x514
#define NRF51_GPIO_REG_DIRSET 0x518
#define NRF51_GPIO_REG_DIRCLR 0x51C
#define NRF51_GPIO_REG_CNF_START 0x700
#define NRF51_GPIO_REG_CNF_END 0x77C
#define NRF51_GPIO_PULLDOWN 1
#define NRF51_GPIO_PULLUP 3
typedef struct NRF51GPIOState {
SysBusDevice parent_obj;
MemoryRegion mmio;
qemu_irq irq;
uint32_t out;
uint32_t in;
uint32_t in_mask;
uint32_t dir;
uint32_t cnf[NRF51_GPIO_PINS];
uint32_t old_out;
uint32_t old_out_connected;
qemu_irq output[NRF51_GPIO_PINS];
} NRF51GPIOState;
#endif