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 139
    rc = pcie_aer_init(d, PCI_ERR_VER, IOH_EP_AER_OFFSET,
                       PCI_ERR_SIZEOF, &err);
I
Isaku Yamahata 已提交
140
    if (rc < 0) {
141
        error_report_err(err);
I
Isaku Yamahata 已提交
142 143 144 145
        goto err;
    }
    pcie_aer_root_init(d);
    ioh3420_aer_vector_update(d);
C
Cao jin 已提交
146

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

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

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

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

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

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

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

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

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

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