151 lines
5.0 KiB
C
151 lines
5.0 KiB
C
/*
|
|
* Copyright (C) 2020 Intel Corporation.
|
|
*
|
|
* SPDX-License-Identifier: BSD-3-Clause
|
|
*/
|
|
|
|
|
|
/*
|
|
* Emulate GPIO registers which are only accessible through Primary to Sideband Bridge (P2SB).
|
|
*
|
|
* Intercept accesses to MISCCFG.GPDMINTSEL[31:24] and PADCFG1.INTSEL[7:0] GPIO registers which hold physical interrupt
|
|
* lines and return virtualized values upon read in accordance with the gsi to vgsi mappings given by the VM config.
|
|
*
|
|
* P2SB_BAR_ADDR: 0xFD000000 (fixed by BIOS)
|
|
*
|
|
*
|
|
* --------------------------------------------------------------------------------------
|
|
* SideBand Endpoint Name | Port ID
|
|
* --------------------------------------------------------------------------------------
|
|
* GPIO Community 5 | 0x69
|
|
* GPIO Community 4 | 0x6A
|
|
* GPIO Community 3 | 0x6B
|
|
* GPIO Community 2 | 0x6C
|
|
* GPIO Community 1 | 0x6D
|
|
* GPIO Community 0 | 0x6E
|
|
* --------------------------------------------------------------------------------------
|
|
*
|
|
* Private Configuration Register (PCR) Address = P2SB_BAR_ADDR + (Port ID << 16) + Register Offset
|
|
* e.g.)
|
|
* GPIO_COMMUNITY_5_PCR_BASE = P2SB_BAR_ADDR + (0x69 << 16) = 0xFD690000
|
|
* GPIO_COMMUNITY_5_MISCCFG = GPIO_COMMUNITY_5_PCR_BASE + 0x010 = 0xFD690010
|
|
* GPIO_COMMUNITY_5_PAD0_CFG1 = GPIO_COMMUNITY_5_PCR_BASE + 0x704 = 0xFD690704
|
|
* GPIO_COMMUNITY_5_PAD1_CFG1 = GPIO_COMMUNITY_5_PCR_BASE + 0x714 = 0xFD690714
|
|
* GPIO_COMMUNITY_5_PAD2_CFG1 = GPIO_COMMUNITY_5_PCR_BASE + 0x724 = 0xFD690724
|
|
* ....
|
|
*
|
|
*/
|
|
|
|
#include <types.h>
|
|
#include <errno.h>
|
|
#include <asm/guest/vm.h>
|
|
#include <asm/guest/ept.h>
|
|
#include <asm/guest/assign.h>
|
|
#include <asm/io.h>
|
|
#include <asm/mmu.h>
|
|
|
|
#ifdef P2SB_VGPIO_DM_ENABLED
|
|
|
|
#define P2SB_PORTID_SHIFT 16U
|
|
#define P2SB_AGENT_NUM 256U
|
|
#define P2SB_PCR_SPACE_SIZE_PER_AGENT 0x10000U
|
|
#define P2SB_PCR_SPACE_SIZE_TOTAL (P2SB_AGENT_NUM * P2SB_PCR_SPACE_SIZE_PER_AGENT)
|
|
#define P2SB_PCR_SPACE_MASK ((1UL << P2SB_PORTID_SHIFT) - 1UL)
|
|
|
|
#define GPIO_MISCCFG 0x010U
|
|
#define GPIO_MISGCFG_GPDMINTSEL_SHIFT 24U
|
|
#define GPIO_PADBAR 0x00CU
|
|
#define GPIO_PADCFG1 0x004U
|
|
#define GPIO_PADCFG1_INTSEL_SHIFT 0U
|
|
|
|
#define GPIO_INVALID_PIN 0xFFU
|
|
|
|
/**
|
|
* @return vpin mapped to the given phys_pin in accordance with the VM config, if not found return 0xFF as invalid pin
|
|
*/
|
|
static uint32_t ioapic_pin_to_vpin(struct acrn_vm *vm, const struct acrn_vm_config *vm_config, const uint32_t phys_pin)
|
|
{
|
|
uint32_t i;
|
|
uint32_t vpin = GPIO_INVALID_PIN;
|
|
struct acrn_single_vioapic *vioapic;
|
|
|
|
for (i = 0U; i < vm_config->pt_intx_num; i++) {
|
|
if (phys_pin == gsi_to_ioapic_pin(vm_config->pt_intx[i].phys_gsi)) {
|
|
vioapic = vgsi_to_vioapic_and_vpin(vm, vm_config->pt_intx[i].virt_gsi, &vpin);
|
|
if (!vioapic) {
|
|
vpin = GPIO_INVALID_PIN;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
return vpin;
|
|
}
|
|
|
|
static int32_t vgpio_mmio_handler(struct io_request *io_req, void *data)
|
|
{
|
|
struct acrn_mmio_request *mmio = &io_req->reqs.mmio_request;
|
|
struct acrn_vm *vm = (struct acrn_vm *) data;
|
|
struct acrn_vm_config *vm_config = get_vm_config(vm->vm_id);
|
|
int32_t ret = 0;
|
|
|
|
uint64_t hpa = P2SB_BAR_ADDR + (mmio->address & (uint64_t)(P2SB_PCR_SPACE_SIZE_TOTAL - 1));
|
|
void *hva = hpa2hva(hpa);
|
|
uint64_t reg_offset = hpa & P2SB_PCR_SPACE_MASK;
|
|
|
|
uint32_t value, shift;
|
|
uint32_t padbar, pad0;
|
|
uint32_t phys_pin, virt_pin;
|
|
|
|
/* all gpio registers have 4 bytes size */
|
|
if (mmio->size == 4U) {
|
|
if (mmio->direction == ACRN_IOREQ_DIR_READ) {
|
|
padbar = mmio_read32((const void *)hpa2hva((hpa & ~P2SB_PCR_SPACE_MASK) + GPIO_PADBAR));
|
|
pad0 = padbar & P2SB_PCR_SPACE_MASK;
|
|
value = mmio_read32((const void *)hva);
|
|
|
|
if ((reg_offset == GPIO_MISCCFG) ||
|
|
((reg_offset >= pad0) && ((reg_offset & 0x0FU) == GPIO_PADCFG1))) {
|
|
shift = (reg_offset == GPIO_MISCCFG) ? GPIO_MISGCFG_GPDMINTSEL_SHIFT :
|
|
GPIO_PADCFG1_INTSEL_SHIFT;
|
|
phys_pin = (value >> shift) & 0xFFU;
|
|
virt_pin = ioapic_pin_to_vpin(vm, vm_config, phys_pin);
|
|
value = (value & ~(0xFFU << shift)) | (virt_pin << shift);
|
|
}
|
|
mmio->value = (uint64_t)value;
|
|
} else {
|
|
value = (uint32_t)mmio->value;
|
|
if (reg_offset == GPIO_MISCCFG) { /* discard writes to MISCCFG.GPDMINTSEL[31:24] */
|
|
value = (value & ~(0xFFU << GPIO_MISGCFG_GPDMINTSEL_SHIFT)) |
|
|
(mmio_read32((const void *)hva) & (0xFFU << GPIO_MISGCFG_GPDMINTSEL_SHIFT));
|
|
}
|
|
mmio_write32(value, (void *)hva);
|
|
}
|
|
} else {
|
|
ret = -EINVAL;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @pre vm != NULL && res != NULL
|
|
*/
|
|
void register_vgpio_handler(struct acrn_vm *vm, const struct acrn_mmiores *res)
|
|
{
|
|
uint64_t gpa_start, gpa_end, gpio_pcr_sz;
|
|
uint64_t base_hpa;
|
|
|
|
gpa_start = res->user_vm_pa + (P2SB_BASE_GPIO_PORT_ID << P2SB_PORTID_SHIFT);
|
|
gpio_pcr_sz = P2SB_PCR_SPACE_SIZE_PER_AGENT * P2SB_MAX_GPIO_COMMUNITIES;
|
|
gpa_end = gpa_start + gpio_pcr_sz;
|
|
base_hpa = res->host_pa + (P2SB_BASE_GPIO_PORT_ID << P2SB_PORTID_SHIFT);
|
|
|
|
/* emulate MMIO access to the GPIO private configuration space registers */
|
|
set_paging_supervisor((uint64_t)hpa2hva(base_hpa), gpio_pcr_sz);
|
|
register_mmio_emulation_handler(vm, vgpio_mmio_handler, gpa_start, gpa_end, (void *)vm, false);
|
|
ept_del_mr(vm, (uint64_t *)vm->arch_vm.nworld_eptp, gpa_start, gpio_pcr_sz);
|
|
}
|
|
|
|
#endif
|