80dcd37feb
The ITS code has to check whether various parameters passed in commands are in-bounds, where the limit is defined in terms of the number of bits that are available for the parameter. (For example, the GITS_TYPER.Devbits ID register field specifies the number of DeviceID bits minus 1, and device IDs passed in the MAPTI and MAPD command packets must fit in that many bits.) Currently we have off-by-one bugs in many of these bounds checks. The typical problem is that we define a max_foo as 1 << n. In the Devbits example, we set s->dt.max_ids = 1UL << (GITS_TYPER.Devbits + 1). However later when we do the bounds check we write if (devid > s->dt.max_ids) { /* command error */ } which incorrectly permits a devid of 1 << n. These bugs will not cause QEMU crashes because the ID values being checked are only used for accesses into tables held in guest memory which we access with address_space_*() functions, but they are incorrect behaviour of our emulation. Fix them by standardizing on this pattern: * bounds limits are named num_foos and are the 2^n value (equal to the number of valid foo values) * bounds checks are either if (fooid < num_foos) { good } or if (fooid >= num_foos) { bad } In this commit we fix the handling of the number of IDs in the device table and the collection table, and the number of commands that will fit in the command queue. Signed-off-by: Peter Maydell <peter.maydell@linaro.org> Reviewed-by: Alex Bennée <alex.bennee@linaro.org>
111 lines
2.8 KiB
C
111 lines
2.8 KiB
C
/*
|
|
* ITS support for ARM GICv3
|
|
*
|
|
* Copyright (c) 2015 Samsung Electronics Co., Ltd.
|
|
* Written by Pavel Fedin
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifndef QEMU_ARM_GICV3_ITS_COMMON_H
|
|
#define QEMU_ARM_GICV3_ITS_COMMON_H
|
|
|
|
#include "hw/sysbus.h"
|
|
#include "hw/intc/arm_gicv3_common.h"
|
|
#include "qom/object.h"
|
|
|
|
#define TYPE_ARM_GICV3_ITS "arm-gicv3-its"
|
|
|
|
#define ITS_CONTROL_SIZE 0x10000
|
|
#define ITS_TRANS_SIZE 0x10000
|
|
#define ITS_SIZE (ITS_CONTROL_SIZE + ITS_TRANS_SIZE)
|
|
|
|
#define GITS_CTLR 0x0
|
|
#define GITS_IIDR 0x4
|
|
#define GITS_TYPER 0x8
|
|
#define GITS_CBASER 0x80
|
|
#define GITS_CWRITER 0x88
|
|
#define GITS_CREADR 0x90
|
|
#define GITS_BASER 0x100
|
|
|
|
#define GITS_TRANSLATER 0x0040
|
|
|
|
typedef struct {
|
|
bool valid;
|
|
bool indirect;
|
|
uint16_t entry_sz;
|
|
uint32_t page_sz;
|
|
uint32_t num_entries;
|
|
uint32_t num_ids;
|
|
uint64_t base_addr;
|
|
} TableDesc;
|
|
|
|
typedef struct {
|
|
bool valid;
|
|
uint32_t num_entries;
|
|
uint64_t base_addr;
|
|
} CmdQDesc;
|
|
|
|
struct GICv3ITSState {
|
|
SysBusDevice parent_obj;
|
|
|
|
MemoryRegion iomem_main;
|
|
MemoryRegion iomem_its_cntrl;
|
|
MemoryRegion iomem_its_translation;
|
|
|
|
GICv3State *gicv3;
|
|
|
|
int dev_fd; /* kvm device fd if backed by kvm vgic support */
|
|
uint64_t gits_translater_gpa;
|
|
bool translater_gpa_known;
|
|
|
|
/* Registers */
|
|
uint32_t ctlr;
|
|
uint32_t iidr;
|
|
uint64_t typer;
|
|
uint64_t cbaser;
|
|
uint64_t cwriter;
|
|
uint64_t creadr;
|
|
uint64_t baser[8];
|
|
|
|
TableDesc dt;
|
|
TableDesc ct;
|
|
CmdQDesc cq;
|
|
|
|
Error *migration_blocker;
|
|
};
|
|
|
|
typedef struct GICv3ITSState GICv3ITSState;
|
|
|
|
void gicv3_its_init_mmio(GICv3ITSState *s, const MemoryRegionOps *ops,
|
|
const MemoryRegionOps *tops);
|
|
|
|
#define TYPE_ARM_GICV3_ITS_COMMON "arm-gicv3-its-common"
|
|
typedef struct GICv3ITSCommonClass GICv3ITSCommonClass;
|
|
DECLARE_OBJ_CHECKERS(GICv3ITSState, GICv3ITSCommonClass,
|
|
ARM_GICV3_ITS_COMMON, TYPE_ARM_GICV3_ITS_COMMON)
|
|
|
|
struct GICv3ITSCommonClass {
|
|
/*< private >*/
|
|
SysBusDeviceClass parent_class;
|
|
/*< public >*/
|
|
|
|
int (*send_msi)(GICv3ITSState *s, uint32_t data, uint16_t devid);
|
|
void (*pre_save)(GICv3ITSState *s);
|
|
void (*post_load)(GICv3ITSState *s);
|
|
};
|
|
|
|
|
|
#endif
|