ioh3420.c 6.5 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
/*
 * ioh3420.c
 * Intel X58 north bridge IOH
 * PCI Express root port device id 3420
 *
 * Copyright (c) 2010 Isaku Yamahata <yamahata at valinux co jp>
 *                    VA Linux Systems Japan K.K.
 *
 * 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/>.
 */

23 24 25
#include "hw/pci/pci_ids.h"
#include "hw/pci/msi.h"
#include "hw/pci/pcie.h"
26
#include "ioh3420.h"
27 28 29 30 31 32 33 34 35 36 37 38

#define PCI_DEVICE_ID_IOH_EPORT         0x3420  /* D0:F0 express mode */
#define PCI_DEVICE_ID_IOH_REV           0x2
#define IOH_EP_SSVID_OFFSET             0x40
#define IOH_EP_SSVID_SVID               PCI_VENDOR_ID_INTEL
#define IOH_EP_SSVID_SSID               0
#define IOH_EP_MSI_OFFSET               0x60
#define IOH_EP_MSI_SUPPORTED_FLAGS      PCI_MSI_FLAGS_MASKBIT
#define IOH_EP_MSI_NR_VECTOR            2
#define IOH_EP_EXP_OFFSET               0x90
#define IOH_EP_AER_OFFSET               0x100

I
Isaku Yamahata 已提交
39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
/*
 * If two MSI vector are allocated, Advanced Error Interrupt Message Number
 * is 1. otherwise 0.
 * 17.12.5.10 RPERRSTS,  32:27 bit Advanced Error Interrupt Message Number.
 */
static uint8_t ioh3420_aer_vector(const PCIDevice *d)
{
    switch (msi_nr_vectors_allocated(d)) {
    case 1:
        return 0;
    case 2:
        return 1;
    case 4:
    case 8:
    case 16:
    case 32:
    default:
        break;
    }
    abort();
    return 0;
}

static void ioh3420_aer_vector_update(PCIDevice *d)
{
    pcie_aer_root_set_vector(d, ioh3420_aer_vector(d));
}

67 68 69
static void ioh3420_write_config(PCIDevice *d,
                                   uint32_t address, uint32_t val, int len)
{
I
Isaku Yamahata 已提交
70 71 72
    uint32_t root_cmd =
        pci_get_long(d->config + d->exp.aer_cap + PCI_ERR_ROOT_COMMAND);

73
    pci_bridge_write_config(d, address, val, len);
I
Isaku Yamahata 已提交
74
    ioh3420_aer_vector_update(d);
75
    pcie_cap_slot_write_config(d, address, val, len);
I
Isaku Yamahata 已提交
76 77
    pcie_aer_write_config(d, address, val, len);
    pcie_aer_root_write_config(d, address, val, len, root_cmd);
78 79 80 81
}

static void ioh3420_reset(DeviceState *qdev)
{
82
    PCIDevice *d = PCI_DEVICE(qdev);
83

I
Isaku Yamahata 已提交
84
    ioh3420_aer_vector_update(d);
85 86 87
    pcie_cap_root_reset(d);
    pcie_cap_deverr_reset(d);
    pcie_cap_slot_reset(d);
I
Isaku Yamahata 已提交
88
    pcie_aer_root_reset(d);
89 90 91 92 93 94
    pci_bridge_reset(qdev);
    pci_bridge_disable_base_limit(d);
}

static int ioh3420_initfn(PCIDevice *d)
{
95 96
    PCIEPort *p = PCIE_PORT(d);
    PCIESlot *s = PCIE_SLOT(d);
97 98
    int rc;

99
    rc = pci_bridge_initfn(d, TYPE_PCIE_BUS);
100 101 102 103 104 105 106 107 108
    if (rc < 0) {
        return rc;
    }

    pcie_port_init_reg(d);

    rc = pci_bridge_ssvid_init(d, IOH_EP_SSVID_OFFSET,
                               IOH_EP_SSVID_SVID, IOH_EP_SSVID_SSID);
    if (rc < 0) {
I
Isaku Yamahata 已提交
109
        goto err_bridge;
110 111 112 113 114
    }
    rc = msi_init(d, IOH_EP_MSI_OFFSET, IOH_EP_MSI_NR_VECTOR,
                  IOH_EP_MSI_SUPPORTED_FLAGS & PCI_MSI_FLAGS_64BIT,
                  IOH_EP_MSI_SUPPORTED_FLAGS & PCI_MSI_FLAGS_MASKBIT);
    if (rc < 0) {
I
Isaku Yamahata 已提交
115
        goto err_bridge;
116 117 118
    }
    rc = pcie_cap_init(d, IOH_EP_EXP_OFFSET, PCI_EXP_TYPE_ROOT_PORT, p->port);
    if (rc < 0) {
I
Isaku Yamahata 已提交
119
        goto err_msi;
120 121 122 123 124 125
    }
    pcie_cap_deverr_init(d);
    pcie_cap_slot_init(d, s->slot);
    pcie_chassis_create(s->chassis);
    rc = pcie_chassis_add_slot(s);
    if (rc < 0) {
I
Isaku Yamahata 已提交
126
        goto err_pcie_cap;
127 128
    }
    pcie_cap_root_init(d);
I
Isaku Yamahata 已提交
129 130 131 132 133 134
    rc = pcie_aer_init(d, IOH_EP_AER_OFFSET);
    if (rc < 0) {
        goto err;
    }
    pcie_aer_root_init(d);
    ioh3420_aer_vector_update(d);
135
    return 0;
I
Isaku Yamahata 已提交
136 137 138 139 140 141 142 143

err:
    pcie_chassis_del_slot(s);
err_pcie_cap:
    pcie_cap_exit(d);
err_msi:
    msi_uninit(d);
err_bridge:
144
    pci_bridge_exitfn(d);
I
Isaku Yamahata 已提交
145
    return rc;
146 147
}

148
static void ioh3420_exitfn(PCIDevice *d)
149
{
150
    PCIESlot *s = PCIE_SLOT(d);
I
Isaku Yamahata 已提交
151 152 153

    pcie_aer_exit(d);
    pcie_chassis_del_slot(s);
154
    pcie_cap_exit(d);
I
Isaku Yamahata 已提交
155
    msi_uninit(d);
156
    pci_bridge_exitfn(d);
157 158 159 160 161 162 163 164 165 166 167 168 169 170
}

PCIESlot *ioh3420_init(PCIBus *bus, int devfn, bool multifunction,
                         const char *bus_name, pci_map_irq_fn map_irq,
                         uint8_t port, uint8_t chassis, uint16_t slot)
{
    PCIDevice *d;
    PCIBridge *br;
    DeviceState *qdev;

    d = pci_create_multifunction(bus, devfn, multifunction, "ioh3420");
    if (!d) {
        return NULL;
    }
171
    br = PCI_BRIDGE(d);
172

173
    qdev = DEVICE(d);
174 175 176 177 178 179
    pci_bridge_map_irq(br, bus_name, map_irq);
    qdev_prop_set_uint8(qdev, "port", port);
    qdev_prop_set_uint8(qdev, "chassis", chassis);
    qdev_prop_set_uint16(qdev, "slot", slot);
    qdev_init_nofail(qdev);

180
    return PCIE_SLOT(d);
181 182
}

183 184 185 186 187 188
static Property ioh3420_props[] = {
    DEFINE_PROP_BIT(COMPAT_PROP_PCP, PCIDevice, cap_present,
                    QEMU_PCIE_SLTCAP_PCP_BITNR, true),
    DEFINE_PROP_END_OF_LIST()
};

189 190 191 192
static const VMStateDescription vmstate_ioh3420 = {
    .name = "ioh-3240-express-root-port",
    .version_id = 1,
    .minimum_version_id = 1,
193
    .post_load = pcie_cap_slot_post_load,
194
    .fields = (VMStateField[]) {
195 196 197
        VMSTATE_PCIE_DEVICE(parent_obj.parent_obj.parent_obj, PCIESlot),
        VMSTATE_STRUCT(parent_obj.parent_obj.parent_obj.exp.aer_log,
                       PCIESlot, 0, vmstate_pcie_aer_log, PCIEAERLog),
198 199 200 201
        VMSTATE_END_OF_LIST()
    }
};

202 203
static void ioh3420_class_init(ObjectClass *klass, void *data)
{
204
    DeviceClass *dc = DEVICE_CLASS(klass);
205 206 207 208 209 210 211 212 213 214
    PCIDeviceClass *k = PCI_DEVICE_CLASS(klass);

    k->is_express = 1;
    k->is_bridge = 1;
    k->config_write = ioh3420_write_config;
    k->init = ioh3420_initfn;
    k->exit = ioh3420_exitfn;
    k->vendor_id = PCI_VENDOR_ID_INTEL;
    k->device_id = PCI_DEVICE_ID_IOH_EPORT;
    k->revision = PCI_DEVICE_ID_IOH_REV;
215
    set_bit(DEVICE_CATEGORY_BRIDGE, dc->categories);
216 217 218
    dc->desc = "Intel IOH device id 3420 PCIE Root Port";
    dc->reset = ioh3420_reset;
    dc->vmsd = &vmstate_ioh3420;
219
    dc->props = ioh3420_props;
220 221
}

222
static const TypeInfo ioh3420_info = {
223
    .name          = "ioh3420",
224
    .parent        = TYPE_PCIE_SLOT,
225
    .class_init    = ioh3420_class_init,
226 227
};

A
Andreas Färber 已提交
228
static void ioh3420_register_types(void)
229
{
230
    type_register_static(&ioh3420_info);
231 232
}

A
Andreas Färber 已提交
233
type_init(ioh3420_register_types)
234 235 236 237 238 239 240 241 242

/*
 * Local variables:
 *  c-indent-level: 4
 *  c-basic-offset: 4
 *  tab-width: 8
 *  indent-tab-mode: nil
 * End:
 */