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_bridge_initfn(d, TYPE_PCIE_BUS);
104 105 106 107 108
    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
    }
C
Cao jin 已提交
111

112 113
    rc = msi_init(d, IOH_EP_MSI_OFFSET, IOH_EP_MSI_NR_VECTOR,
                  IOH_EP_MSI_SUPPORTED_FLAGS & PCI_MSI_FLAGS_64BIT,
114
                  IOH_EP_MSI_SUPPORTED_FLAGS & PCI_MSI_FLAGS_MASKBIT, &err);
115
    if (rc < 0) {
116 117
        assert(rc == -ENOTSUP);
        error_report_err(err);
I
Isaku Yamahata 已提交
118
        goto err_bridge;
119
    }
C
Cao jin 已提交
120

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

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

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

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

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

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

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

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

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

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

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

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

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

A
Andreas Färber 已提交
218
type_init(ioh3420_register_types)
219 220 221 222 223 224 225 226 227

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