ioh3420.c 6.0 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/>.
 */

P
Peter Maydell 已提交
23
#include "qemu/osdep.h"
24 25 26
#include "hw/pci/pci_ids.h"
#include "hw/pci/msi.h"
#include "hw/pci/pcie.h"
27
#include "ioh3420.h"
28
#include "qapi/error.h"
29 30 31 32 33 34 35 36 37 38 39 40

#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 已提交
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 67 68
/*
 * 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));
}

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

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

static void ioh3420_reset(DeviceState *qdev)
{
84
    PCIDevice *d = PCI_DEVICE(qdev);
85

I
Isaku Yamahata 已提交
86
    ioh3420_aer_vector_update(d);
87 88 89
    pcie_cap_root_reset(d);
    pcie_cap_deverr_reset(d);
    pcie_cap_slot_reset(d);
K
Knut Omang 已提交
90
    pcie_cap_arifwd_reset(d);
I
Isaku Yamahata 已提交
91
    pcie_aer_root_reset(d);
92 93 94 95 96 97
    pci_bridge_reset(qdev);
    pci_bridge_disable_base_limit(d);
}

static int ioh3420_initfn(PCIDevice *d)
{
98 99
    PCIEPort *p = PCIE_PORT(d);
    PCIESlot *s = PCIE_SLOT(d);
100
    int rc;
101
    Error *err = NULL;
102

103
    pci_config_set_interrupt_pin(d->config, 1);
104
    pci_bridge_initfn(d, TYPE_PCIE_BUS);
105 106 107 108 109
    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 已提交
110
        goto err_bridge;
111
    }
C
Cao jin 已提交
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,
115
                  IOH_EP_MSI_SUPPORTED_FLAGS & PCI_MSI_FLAGS_MASKBIT, &err);
116
    if (rc < 0) {
117 118
        assert(rc == -ENOTSUP);
        error_report_err(err);
I
Isaku Yamahata 已提交
119
        goto err_bridge;
120
    }
C
Cao jin 已提交
121

122 123
    rc = pcie_cap_init(d, IOH_EP_EXP_OFFSET, PCI_EXP_TYPE_ROOT_PORT, p->port);
    if (rc < 0) {
I
Isaku Yamahata 已提交
124
        goto err_msi;
125
    }
126

K
Knut Omang 已提交
127
    pcie_cap_arifwd_init(d);
128 129
    pcie_cap_deverr_init(d);
    pcie_cap_slot_init(d, s->slot);
C
Cao jin 已提交
130 131
    pcie_cap_root_init(d);

132 133 134
    pcie_chassis_create(s->chassis);
    rc = pcie_chassis_add_slot(s);
    if (rc < 0) {
I
Isaku Yamahata 已提交
135
        goto err_pcie_cap;
136
    }
C
Cao jin 已提交
137

138
    rc = pcie_aer_init(d, IOH_EP_AER_OFFSET, PCI_ERR_SIZEOF, &err);
I
Isaku Yamahata 已提交
139
    if (rc < 0) {
140
        error_report_err(err);
I
Isaku Yamahata 已提交
141 142 143 144
        goto err;
    }
    pcie_aer_root_init(d);
    ioh3420_aer_vector_update(d);
C
Cao jin 已提交
145

146
    return 0;
I
Isaku Yamahata 已提交
147 148 149 150 151 152 153 154

err:
    pcie_chassis_del_slot(s);
err_pcie_cap:
    pcie_cap_exit(d);
err_msi:
    msi_uninit(d);
err_bridge:
155
    pci_bridge_exitfn(d);
I
Isaku Yamahata 已提交
156
    return rc;
157 158
}

159
static void ioh3420_exitfn(PCIDevice *d)
160
{
161
    PCIESlot *s = PCIE_SLOT(d);
I
Isaku Yamahata 已提交
162 163 164

    pcie_aer_exit(d);
    pcie_chassis_del_slot(s);
165
    pcie_cap_exit(d);
I
Isaku Yamahata 已提交
166
    msi_uninit(d);
167
    pci_bridge_exitfn(d);
168 169
}

170 171 172 173 174 175
static Property ioh3420_props[] = {
    DEFINE_PROP_BIT(COMPAT_PROP_PCP, PCIDevice, cap_present,
                    QEMU_PCIE_SLTCAP_PCP_BITNR, true),
    DEFINE_PROP_END_OF_LIST()
};

176 177 178 179
static const VMStateDescription vmstate_ioh3420 = {
    .name = "ioh-3240-express-root-port",
    .version_id = 1,
    .minimum_version_id = 1,
180
    .post_load = pcie_cap_slot_post_load,
181
    .fields = (VMStateField[]) {
182 183 184
        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),
185 186 187 188
        VMSTATE_END_OF_LIST()
    }
};

189 190
static void ioh3420_class_init(ObjectClass *klass, void *data)
{
191
    DeviceClass *dc = DEVICE_CLASS(klass);
192 193 194 195 196 197 198 199 200 201
    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;
202
    set_bit(DEVICE_CATEGORY_BRIDGE, dc->categories);
203 204 205
    dc->desc = "Intel IOH device id 3420 PCIE Root Port";
    dc->reset = ioh3420_reset;
    dc->vmsd = &vmstate_ioh3420;
206
    dc->props = ioh3420_props;
207 208
}

209
static const TypeInfo ioh3420_info = {
210
    .name          = "ioh3420",
211
    .parent        = TYPE_PCIE_SLOT,
212
    .class_init    = ioh3420_class_init,
213 214
};

A
Andreas Färber 已提交
215
static void ioh3420_register_types(void)
216
{
217
    type_register_static(&ioh3420_info);
218 219
}

A
Andreas Färber 已提交
220
type_init(ioh3420_register_types)