pci.c 28.0 KB
Newer Older
B
bellard 已提交
1 2 3 4
/*
 * QEMU PCI bus manager
 *
 * Copyright (c) 2004 Fabrice Bellard
5
 *
B
bellard 已提交
6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in
 * all copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 * THE SOFTWARE.
 */
P
pbrook 已提交
24 25
#include "hw.h"
#include "pci.h"
A
aliguori 已提交
26
#include "monitor.h"
P
pbrook 已提交
27
#include "net.h"
28
#include "sysemu.h"
B
bellard 已提交
29 30 31

//#define DEBUG_PCI

32
struct PCIBus {
P
Paul Brook 已提交
33
    BusState qbus;
34 35
    int bus_num;
    int devfn_min;
P
pbrook 已提交
36
    pci_set_irq_fn set_irq;
37
    pci_map_irq_fn map_irq;
38
    uint32_t config_reg; /* XXX: suppress */
39 40
    /* low level pic */
    SetIRQFunc *low_set_irq;
P
pbrook 已提交
41
    qemu_irq *irq_opaque;
42
    PCIDevice *devices[256];
P
pbrook 已提交
43 44
    PCIDevice *parent_dev;
    PCIBus *next;
45 46
    /* The bus IRQ state is the logical OR of the connected devices.
       Keep a count of the number of devices with raised IRQs.  */
47
    int nirq;
P
pbrook 已提交
48
    int irq_count[];
49
};
B
bellard 已提交
50

B
bellard 已提交
51
static void pci_update_mappings(PCIDevice *d);
P
pbrook 已提交
52
static void pci_set_irq(void *opaque, int irq_num, int level);
B
bellard 已提交
53

B
bellard 已提交
54
target_phys_addr_t pci_mem_base;
55 56
static uint16_t pci_default_sub_vendor_id = PCI_SUBVENDOR_ID_REDHAT_QUMRANET;
static uint16_t pci_default_sub_device_id = PCI_SUBDEVICE_ID_QEMU;
57 58
static PCIBus *first_bus;

59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
static void pcibus_save(QEMUFile *f, void *opaque)
{
    PCIBus *bus = (PCIBus *)opaque;
    int i;

    qemu_put_be32(f, bus->nirq);
    for (i = 0; i < bus->nirq; i++)
        qemu_put_be32(f, bus->irq_count[i]);
}

static int  pcibus_load(QEMUFile *f, void *opaque, int version_id)
{
    PCIBus *bus = (PCIBus *)opaque;
    int i, nirq;

    if (version_id != 1)
        return -EINVAL;

    nirq = qemu_get_be32(f);
    if (bus->nirq != nirq) {
        fprintf(stderr, "pcibus_load: nirq mismatch: src=%d dst=%d\n",
                nirq, bus->nirq);
        return -EINVAL;
    }

    for (i = 0; i < nirq; i++)
        bus->irq_count[i] = qemu_get_be32(f);

    return 0;
}

G
Gleb Natapov 已提交
90 91 92 93 94 95 96 97 98 99 100 101 102 103 104
static void pci_bus_reset(void *opaque)
{
    PCIBus *bus = (PCIBus *)opaque;
    int i;

    for (i = 0; i < bus->nirq; i++) {
        bus->irq_count[i] = 0;
    }
    for (i = 0; i < 256; i++) {
        if (bus->devices[i])
            memset(bus->devices[i]->irq_state, 0,
                   sizeof(bus->devices[i]->irq_state));
    }
}

P
Paul Brook 已提交
105 106
PCIBus *pci_register_bus(DeviceState *parent, const char *name,
                         pci_set_irq_fn set_irq, pci_map_irq_fn map_irq,
P
pbrook 已提交
107
                         qemu_irq *pic, int devfn_min, int nirq)
108 109
{
    PCIBus *bus;
110 111
    static int nbus = 0;

P
Paul Brook 已提交
112 113 114
    bus = FROM_QBUS(PCIBus, qbus_create(BUS_TYPE_PCI,
                                        sizeof(PCIBus) + (nirq * sizeof(int)),
                                        parent, name));
P
pbrook 已提交
115
    bus->set_irq = set_irq;
116
    bus->map_irq = map_irq;
P
pbrook 已提交
117 118
    bus->irq_opaque = pic;
    bus->devfn_min = devfn_min;
119
    bus->nirq = nirq;
I
Isaku Yamahata 已提交
120
    bus->next = first_bus;
121
    first_bus = bus;
122
    register_savevm("PCIBUS", nbus++, 1, pcibus_save, pcibus_load, bus);
G
Gleb Natapov 已提交
123
    qemu_register_reset(pci_bus_reset, 0, bus);
124 125
    return bus;
}
B
bellard 已提交
126

127
static PCIBus *pci_register_secondary_bus(PCIDevice *dev, pci_map_irq_fn map_irq)
P
pbrook 已提交
128 129 130 131 132 133 134 135 136 137
{
    PCIBus *bus;
    bus = qemu_mallocz(sizeof(PCIBus));
    bus->map_irq = map_irq;
    bus->parent_dev = dev;
    bus->next = dev->bus->next;
    dev->bus->next = bus;
    return bus;
}

P
pbrook 已提交
138 139 140 141 142
int pci_bus_num(PCIBus *s)
{
    return s->bus_num;
}

B
bellard 已提交
143
void pci_device_save(PCIDevice *s, QEMUFile *f)
144
{
145 146 147
    int i;

    qemu_put_be32(f, 2); /* PCI device version */
148
    qemu_put_buffer(f, s->config, 256);
149 150
    for (i = 0; i < 4; i++)
        qemu_put_be32(f, s->irq_state[i]);
151 152
}

B
bellard 已提交
153
int pci_device_load(PCIDevice *s, QEMUFile *f)
154
{
155
    uint8_t config[PCI_CONFIG_SPACE_SIZE];
B
bellard 已提交
156
    uint32_t version_id;
157 158
    int i;

B
bellard 已提交
159
    version_id = qemu_get_be32(f);
160
    if (version_id > 2)
161
        return -EINVAL;
162 163 164 165 166 167
    qemu_get_buffer(f, config, sizeof config);
    for (i = 0; i < sizeof config; ++i)
        if ((config[i] ^ s->config[i]) & s->cmask[i] & ~s->wmask[i])
            return -EINVAL;
    memcpy(s->config, config, sizeof config);

B
bellard 已提交
168
    pci_update_mappings(s);
169 170 171 172

    if (version_id >= 2)
        for (i = 0; i < 4; i ++)
            s->irq_state[i] = qemu_get_be32(f);
173 174 175
    return 0;
}

176 177 178 179 180 181 182 183 184 185
static int pci_set_default_subsystem_id(PCIDevice *pci_dev)
{
    uint16_t *id;

    id = (void*)(&pci_dev->config[PCI_SUBVENDOR_ID]);
    id[0] = cpu_to_le16(pci_default_sub_vendor_id);
    id[1] = cpu_to_le16(pci_default_sub_device_id);
    return 0;
}

186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234
/*
 * Parse [[<domain>:]<bus>:]<slot>, return -1 on error
 */
static int pci_parse_devaddr(const char *addr, int *domp, int *busp, unsigned *slotp)
{
    const char *p;
    char *e;
    unsigned long val;
    unsigned long dom = 0, bus = 0;
    unsigned slot = 0;

    p = addr;
    val = strtoul(p, &e, 16);
    if (e == p)
	return -1;
    if (*e == ':') {
	bus = val;
	p = e + 1;
	val = strtoul(p, &e, 16);
	if (e == p)
	    return -1;
	if (*e == ':') {
	    dom = bus;
	    bus = val;
	    p = e + 1;
	    val = strtoul(p, &e, 16);
	    if (e == p)
		return -1;
	}
    }

    if (dom > 0xffff || bus > 0xff || val > 0x1f)
	return -1;

    slot = val;

    if (*e)
	return -1;

    /* Note: QEMU doesn't implement domains other than 0 */
    if (dom != 0 || pci_find_bus(bus) == NULL)
	return -1;

    *domp = dom;
    *busp = bus;
    *slotp = slot;
    return 0;
}

235 236
int pci_read_devaddr(Monitor *mon, const char *addr, int *domp, int *busp,
                     unsigned *slotp)
237
{
238 239 240 241 242 243
    /* strip legacy tag */
    if (!strncmp(addr, "pci_addr=", 9)) {
        addr += 9;
    }
    if (pci_parse_devaddr(addr, domp, busp, slotp)) {
        monitor_printf(mon, "Invalid pci address\n");
244
        return -1;
245 246
    }
    return 0;
247 248
}

249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266
static PCIBus *pci_get_bus_devfn(int *devfnp, const char *devaddr)
{
    int dom, bus;
    unsigned slot;

    if (!devaddr) {
        *devfnp = -1;
        return pci_find_bus(0);
    }

    if (pci_parse_devaddr(devaddr, &dom, &bus, &slot) < 0) {
        return NULL;
    }

    *devfnp = slot << 3;
    return pci_find_bus(bus);
}

267 268 269 270 271 272 273 274 275 276 277 278
static void pci_init_cmask(PCIDevice *dev)
{
    pci_set_word(dev->cmask + PCI_VENDOR_ID, 0xffff);
    pci_set_word(dev->cmask + PCI_DEVICE_ID, 0xffff);
    dev->cmask[PCI_STATUS] = PCI_STATUS_CAP_LIST;
    dev->cmask[PCI_REVISION_ID] = 0xff;
    dev->cmask[PCI_CLASS_PROG] = 0xff;
    pci_set_word(dev->cmask + PCI_CLASS_DEVICE, 0xffff);
    dev->cmask[PCI_HEADER_TYPE] = 0xff;
    dev->cmask[PCI_CAPABILITY_LIST] = 0xff;
}

279 280 281 282 283 284 285 286 287 288 289
static void pci_init_wmask(PCIDevice *dev)
{
    int i;
    dev->wmask[PCI_CACHE_LINE_SIZE] = 0xff;
    dev->wmask[PCI_INTERRUPT_LINE] = 0xff;
    dev->wmask[PCI_COMMAND] = PCI_COMMAND_IO | PCI_COMMAND_MEMORY
                              | PCI_COMMAND_MASTER;
    for (i = PCI_CONFIG_HEADER_SIZE; i < PCI_CONFIG_SPACE_SIZE; ++i)
        dev->wmask[i] = 0xff;
}

B
bellard 已提交
290
/* -1 for devfn means auto assign */
P
Paul Brook 已提交
291 292 293 294
static PCIDevice *do_pci_register_device(PCIDevice *pci_dev, PCIBus *bus,
                                         const char *name, int devfn,
                                         PCIConfigReadFunc *config_read,
                                         PCIConfigWriteFunc *config_write)
B
bellard 已提交
295 296
{
    if (devfn < 0) {
297 298
        for(devfn = bus->devfn_min ; devfn < 256; devfn += 8) {
            if (!bus->devices[devfn])
B
bellard 已提交
299 300 301 302
                goto found;
        }
        return NULL;
    found: ;
303 304
    } else if (bus->devices[devfn]) {
        return NULL;
B
bellard 已提交
305
    }
306
    pci_dev->bus = bus;
B
bellard 已提交
307 308
    pci_dev->devfn = devfn;
    pstrcpy(pci_dev->name, sizeof(pci_dev->name), name);
309
    memset(pci_dev->irq_state, 0, sizeof(pci_dev->irq_state));
310
    pci_set_default_subsystem_id(pci_dev);
311
    pci_init_cmask(pci_dev);
312
    pci_init_wmask(pci_dev);
313 314 315 316 317

    if (!config_read)
        config_read = pci_default_read_config;
    if (!config_write)
        config_write = pci_default_write_config;
B
bellard 已提交
318 319
    pci_dev->config_read = config_read;
    pci_dev->config_write = config_write;
320
    bus->devices[devfn] = pci_dev;
P
pbrook 已提交
321
    pci_dev->irq = qemu_allocate_irqs(pci_set_irq, pci_dev, 4);
B
bellard 已提交
322 323 324
    return pci_dev;
}

P
Paul Brook 已提交
325 326 327 328 329 330 331 332 333 334 335 336
PCIDevice *pci_register_device(PCIBus *bus, const char *name,
                               int instance_size, int devfn,
                               PCIConfigReadFunc *config_read,
                               PCIConfigWriteFunc *config_write)
{
    PCIDevice *pci_dev;

    pci_dev = qemu_mallocz(instance_size);
    pci_dev = do_pci_register_device(pci_dev, bus, name, devfn,
                                     config_read, config_write);
    return pci_dev;
}
337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373
static target_phys_addr_t pci_to_cpu_addr(target_phys_addr_t addr)
{
    return addr + pci_mem_base;
}

static void pci_unregister_io_regions(PCIDevice *pci_dev)
{
    PCIIORegion *r;
    int i;

    for(i = 0; i < PCI_NUM_REGIONS; i++) {
        r = &pci_dev->io_regions[i];
        if (!r->size || r->addr == -1)
            continue;
        if (r->type == PCI_ADDRESS_SPACE_IO) {
            isa_unassign_ioport(r->addr, r->size);
        } else {
            cpu_register_physical_memory(pci_to_cpu_addr(r->addr),
                                                     r->size,
                                                     IO_MEM_UNASSIGNED);
        }
    }
}

int pci_unregister_device(PCIDevice *pci_dev)
{
    int ret = 0;

    if (pci_dev->unregister)
        ret = pci_dev->unregister(pci_dev);
    if (ret)
        return ret;

    pci_unregister_io_regions(pci_dev);

    qemu_free_irqs(pci_dev->irq);
    pci_dev->bus->devices[pci_dev->devfn] = NULL;
P
Paul Brook 已提交
374
    qdev_free(&pci_dev->qdev);
375 376 377
    return 0;
}

378
void pci_register_bar(PCIDevice *pci_dev, int region_num,
379
                            uint32_t size, int type,
B
bellard 已提交
380 381 382
                            PCIMapIORegionFunc *map_func)
{
    PCIIORegion *r;
P
pbrook 已提交
383
    uint32_t addr;
384
    uint32_t wmask;
B
bellard 已提交
385

386
    if ((unsigned int)region_num >= PCI_NUM_REGIONS)
B
bellard 已提交
387
        return;
388 389 390 391 392 393 394

    if (size & (size-1)) {
        fprintf(stderr, "ERROR: PCI region size must be pow2 "
                    "type=0x%x, size=0x%x\n", type, size);
        exit(1);
    }

B
bellard 已提交
395 396 397 398 399
    r = &pci_dev->io_regions[region_num];
    r->addr = -1;
    r->size = size;
    r->type = type;
    r->map_func = map_func;
400 401

    wmask = ~(size - 1);
P
pbrook 已提交
402 403
    if (region_num == PCI_ROM_SLOT) {
        addr = 0x30;
404 405
        /* ROM enable bit is writeable */
        wmask |= 1;
P
pbrook 已提交
406 407 408 409
    } else {
        addr = 0x10 + region_num * 4;
    }
    *(uint32_t *)(pci_dev->config + addr) = cpu_to_le32(type);
410
    *(uint32_t *)(pci_dev->wmask + addr) = cpu_to_le32(wmask);
411
    *(uint32_t *)(pci_dev->cmask + addr) = 0xffffffff;
B
bellard 已提交
412 413
}

414 415 416 417
static void pci_update_mappings(PCIDevice *d)
{
    PCIIORegion *r;
    int cmd, i;
418
    uint32_t last_addr, new_addr, config_ofs;
419

420
    cmd = le16_to_cpu(*(uint16_t *)(d->config + PCI_COMMAND));
421
    for(i = 0; i < PCI_NUM_REGIONS; i++) {
422
        r = &d->io_regions[i];
423 424 425 426 427
        if (i == PCI_ROM_SLOT) {
            config_ofs = 0x30;
        } else {
            config_ofs = 0x10 + i * 4;
        }
428 429 430
        if (r->size != 0) {
            if (r->type & PCI_ADDRESS_SPACE_IO) {
                if (cmd & PCI_COMMAND_IO) {
431
                    new_addr = le32_to_cpu(*(uint32_t *)(d->config +
432
                                                         config_ofs));
433 434 435 436 437 438 439 440 441 442 443 444
                    new_addr = new_addr & ~(r->size - 1);
                    last_addr = new_addr + r->size - 1;
                    /* NOTE: we have only 64K ioports on PC */
                    if (last_addr <= new_addr || new_addr == 0 ||
                        last_addr >= 0x10000) {
                        new_addr = -1;
                    }
                } else {
                    new_addr = -1;
                }
            } else {
                if (cmd & PCI_COMMAND_MEMORY) {
445
                    new_addr = le32_to_cpu(*(uint32_t *)(d->config +
446 447 448 449
                                                         config_ofs));
                    /* the ROM slot has a specific enable bit */
                    if (i == PCI_ROM_SLOT && !(new_addr & 1))
                        goto no_mem_map;
450 451 452 453 454 455 456 457 458 459 460
                    new_addr = new_addr & ~(r->size - 1);
                    last_addr = new_addr + r->size - 1;
                    /* NOTE: we do not support wrapping */
                    /* XXX: as we cannot support really dynamic
                       mappings, we handle specific values as invalid
                       mappings. */
                    if (last_addr <= new_addr || new_addr == 0 ||
                        last_addr == -1) {
                        new_addr = -1;
                    }
                } else {
461
                no_mem_map:
462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478
                    new_addr = -1;
                }
            }
            /* now do the real mapping */
            if (new_addr != r->addr) {
                if (r->addr != -1) {
                    if (r->type & PCI_ADDRESS_SPACE_IO) {
                        int class;
                        /* NOTE: specific hack for IDE in PC case:
                           only one byte must be mapped. */
                        class = d->config[0x0a] | (d->config[0x0b] << 8);
                        if (class == 0x0101 && r->size == 4) {
                            isa_unassign_ioport(r->addr + 2, 1);
                        } else {
                            isa_unassign_ioport(r->addr, r->size);
                        }
                    } else {
P
pbrook 已提交
479
                        cpu_register_physical_memory(pci_to_cpu_addr(r->addr),
480
                                                     r->size,
481
                                                     IO_MEM_UNASSIGNED);
A
aliguori 已提交
482
                        qemu_unregister_coalesced_mmio(r->addr, r->size);
483 484 485 486 487 488 489 490 491 492 493
                    }
                }
                r->addr = new_addr;
                if (r->addr != -1) {
                    r->map_func(d, i, r->addr, r->size, r->type);
                }
            }
        }
    }
}

494
uint32_t pci_default_read_config(PCIDevice *d,
495
                                 uint32_t address, int len)
B
bellard 已提交
496
{
497
    uint32_t val;
498

499 500 501
    switch(len) {
    default:
    case 4:
502 503 504 505 506 507 508 509 510 511 512 513 514
	if (address <= 0xfc) {
	    val = le32_to_cpu(*(uint32_t *)(d->config + address));
	    break;
	}
	/* fall through */
    case 2:
        if (address <= 0xfe) {
	    val = le16_to_cpu(*(uint16_t *)(d->config + address));
	    break;
	}
	/* fall through */
    case 1:
        val = d->config[address];
515 516 517 518 519
        break;
    }
    return val;
}

520
void pci_default_write_config(PCIDevice *d, uint32_t addr, uint32_t val, int l)
521
{
522 523
    uint8_t orig[PCI_CONFIG_SPACE_SIZE];
    int i;
524 525

    /* not efficient, but simple */
526 527 528 529
    memcpy(orig, d->config, PCI_CONFIG_SPACE_SIZE);
    for(i = 0; i < l && addr < PCI_CONFIG_SPACE_SIZE; val >>= 8, ++i, ++addr) {
        uint8_t wmask = d->wmask[addr];
        d->config[addr] = (d->config[addr] & ~wmask) | (val & wmask);
530
    }
531 532 533
    if (memcmp(orig + PCI_BASE_ADDRESS_0, d->config + PCI_BASE_ADDRESS_0, 24)
        || ((orig[PCI_COMMAND] ^ d->config[PCI_COMMAND])
            & (PCI_COMMAND_MEMORY | PCI_COMMAND_IO)))
534
        pci_update_mappings(d);
B
bellard 已提交
535 536
}

P
pbrook 已提交
537
void pci_data_write(void *opaque, uint32_t addr, uint32_t val, int len)
B
bellard 已提交
538
{
539 540 541
    PCIBus *s = opaque;
    PCIDevice *pci_dev;
    int config_addr, bus_num;
542

B
bellard 已提交
543 544
#if defined(DEBUG_PCI) && 0
    printf("pci_data_write: addr=%08x val=%08x len=%d\n",
P
pbrook 已提交
545
           addr, val, len);
B
bellard 已提交
546
#endif
P
pbrook 已提交
547
    bus_num = (addr >> 16) & 0xff;
P
pbrook 已提交
548 549 550
    while (s && s->bus_num != bus_num)
        s = s->next;
    if (!s)
B
bellard 已提交
551
        return;
P
pbrook 已提交
552
    pci_dev = s->devices[(addr >> 8) & 0xff];
B
bellard 已提交
553 554
    if (!pci_dev)
        return;
P
pbrook 已提交
555
    config_addr = addr & 0xff;
B
bellard 已提交
556 557 558 559
#if defined(DEBUG_PCI)
    printf("pci_config_write: %s: addr=%02x val=%08x len=%d\n",
           pci_dev->name, config_addr, val, len);
#endif
560
    pci_dev->config_write(pci_dev, config_addr, val, len);
B
bellard 已提交
561 562
}

P
pbrook 已提交
563
uint32_t pci_data_read(void *opaque, uint32_t addr, int len)
B
bellard 已提交
564
{
565 566 567
    PCIBus *s = opaque;
    PCIDevice *pci_dev;
    int config_addr, bus_num;
B
bellard 已提交
568 569
    uint32_t val;

P
pbrook 已提交
570
    bus_num = (addr >> 16) & 0xff;
P
pbrook 已提交
571 572 573
    while (s && s->bus_num != bus_num)
        s= s->next;
    if (!s)
B
bellard 已提交
574
        goto fail;
P
pbrook 已提交
575
    pci_dev = s->devices[(addr >> 8) & 0xff];
B
bellard 已提交
576 577
    if (!pci_dev) {
    fail:
578 579 580 581 582 583 584 585 586 587 588 589
        switch(len) {
        case 1:
            val = 0xff;
            break;
        case 2:
            val = 0xffff;
            break;
        default:
        case 4:
            val = 0xffffffff;
            break;
        }
B
bellard 已提交
590 591
        goto the_end;
    }
P
pbrook 已提交
592
    config_addr = addr & 0xff;
B
bellard 已提交
593 594 595 596 597 598 599 600
    val = pci_dev->config_read(pci_dev, config_addr, len);
#if defined(DEBUG_PCI)
    printf("pci_config_read: %s: addr=%02x val=%08x len=%d\n",
           pci_dev->name, config_addr, val, len);
#endif
 the_end:
#if defined(DEBUG_PCI) && 0
    printf("pci_data_read: addr=%08x val=%08x len=%d\n",
P
pbrook 已提交
601
           addr, val, len);
B
bellard 已提交
602 603 604 605
#endif
    return val;
}

P
pbrook 已提交
606 607
/***********************************************************/
/* generic PCI irq support */
608

P
pbrook 已提交
609
/* 0 <= irq_num <= 3. level must be 0 or 1 */
P
pbrook 已提交
610
static void pci_set_irq(void *opaque, int irq_num, int level)
B
bellard 已提交
611
{
P
pbrook 已提交
612
    PCIDevice *pci_dev = (PCIDevice *)opaque;
P
pbrook 已提交
613 614
    PCIBus *bus;
    int change;
615

P
pbrook 已提交
616 617 618
    change = level - pci_dev->irq_state[irq_num];
    if (!change)
        return;
619 620

    pci_dev->irq_state[irq_num] = level;
P
pbrook 已提交
621 622
    for (;;) {
        bus = pci_dev->bus;
P
pbrook 已提交
623
        irq_num = bus->map_irq(pci_dev, irq_num);
P
pbrook 已提交
624 625
        if (bus->set_irq)
            break;
P
pbrook 已提交
626 627 628
        pci_dev = bus->parent_dev;
    }
    bus->irq_count[irq_num] += change;
629
    bus->set_irq(bus->irq_opaque, irq_num, bus->irq_count[irq_num] != 0);
B
bellard 已提交
630 631
}

P
pbrook 已提交
632 633
/***********************************************************/
/* monitor info on PCI */
634

635 636 637 638 639
typedef struct {
    uint16_t class;
    const char *desc;
} pci_class_desc;

640
static const pci_class_desc pci_class_descriptions[] =
641
{
P
pbrook 已提交
642
    { 0x0100, "SCSI controller"},
643
    { 0x0101, "IDE controller"},
T
ths 已提交
644 645 646 647 648 649
    { 0x0102, "Floppy controller"},
    { 0x0103, "IPI controller"},
    { 0x0104, "RAID controller"},
    { 0x0106, "SATA controller"},
    { 0x0107, "SAS controller"},
    { 0x0180, "Storage controller"},
650
    { 0x0200, "Ethernet controller"},
T
ths 已提交
651 652 653 654
    { 0x0201, "Token Ring controller"},
    { 0x0202, "FDDI controller"},
    { 0x0203, "ATM controller"},
    { 0x0280, "Network controller"},
655
    { 0x0300, "VGA controller"},
T
ths 已提交
656 657 658 659 660 661 662 663 664 665
    { 0x0301, "XGA controller"},
    { 0x0302, "3D controller"},
    { 0x0380, "Display controller"},
    { 0x0400, "Video controller"},
    { 0x0401, "Audio controller"},
    { 0x0402, "Phone"},
    { 0x0480, "Multimedia controller"},
    { 0x0500, "RAM controller"},
    { 0x0501, "Flash controller"},
    { 0x0580, "Memory controller"},
666 667
    { 0x0600, "Host bridge"},
    { 0x0601, "ISA bridge"},
T
ths 已提交
668 669
    { 0x0602, "EISA bridge"},
    { 0x0603, "MC bridge"},
670
    { 0x0604, "PCI bridge"},
T
ths 已提交
671 672 673 674 675
    { 0x0605, "PCMCIA bridge"},
    { 0x0606, "NUBUS bridge"},
    { 0x0607, "CARDBUS bridge"},
    { 0x0608, "RACEWAY bridge"},
    { 0x0680, "Bridge"},
676 677 678 679
    { 0x0c03, "USB controller"},
    { 0, NULL}
};

P
pbrook 已提交
680
static void pci_info_device(PCIDevice *d)
681
{
A
aliguori 已提交
682
    Monitor *mon = cur_mon;
P
pbrook 已提交
683 684
    int i, class;
    PCIIORegion *r;
685
    const pci_class_desc *desc;
686

A
aliguori 已提交
687 688
    monitor_printf(mon, "  Bus %2d, device %3d, function %d:\n",
                   d->bus->bus_num, d->devfn >> 3, d->devfn & 7);
P
pbrook 已提交
689
    class = le16_to_cpu(*((uint16_t *)(d->config + PCI_CLASS_DEVICE)));
A
aliguori 已提交
690
    monitor_printf(mon, "    ");
691 692 693 694
    desc = pci_class_descriptions;
    while (desc->desc && class != desc->class)
        desc++;
    if (desc->desc) {
A
aliguori 已提交
695
        monitor_printf(mon, "%s", desc->desc);
696
    } else {
A
aliguori 已提交
697
        monitor_printf(mon, "Class %04x", class);
698
    }
A
aliguori 已提交
699
    monitor_printf(mon, ": PCI device %04x:%04x\n",
P
pbrook 已提交
700 701
           le16_to_cpu(*((uint16_t *)(d->config + PCI_VENDOR_ID))),
           le16_to_cpu(*((uint16_t *)(d->config + PCI_DEVICE_ID))));
702

P
pbrook 已提交
703
    if (d->config[PCI_INTERRUPT_PIN] != 0) {
A
aliguori 已提交
704 705
        monitor_printf(mon, "      IRQ %d.\n",
                       d->config[PCI_INTERRUPT_LINE]);
706
    }
P
pbrook 已提交
707
    if (class == 0x0604) {
A
aliguori 已提交
708
        monitor_printf(mon, "      BUS %d.\n", d->config[0x19]);
P
pbrook 已提交
709
    }
P
pbrook 已提交
710 711 712
    for(i = 0;i < PCI_NUM_REGIONS; i++) {
        r = &d->io_regions[i];
        if (r->size != 0) {
A
aliguori 已提交
713
            monitor_printf(mon, "      BAR%d: ", i);
P
pbrook 已提交
714
            if (r->type & PCI_ADDRESS_SPACE_IO) {
A
aliguori 已提交
715 716
                monitor_printf(mon, "I/O at 0x%04x [0x%04x].\n",
                               r->addr, r->addr + r->size - 1);
P
pbrook 已提交
717
            } else {
A
aliguori 已提交
718 719
                monitor_printf(mon, "32 bit memory at 0x%08x [0x%08x].\n",
                               r->addr, r->addr + r->size - 1);
P
pbrook 已提交
720 721
            }
        }
B
bellard 已提交
722
    }
P
pbrook 已提交
723 724 725
    if (class == 0x0604 && d->config[0x19] != 0) {
        pci_for_each_device(d->config[0x19], pci_info_device);
    }
726 727
}

P
pbrook 已提交
728
void pci_for_each_device(int bus_num, void (*fn)(PCIDevice *d))
729
{
P
pbrook 已提交
730
    PCIBus *bus = first_bus;
731
    PCIDevice *d;
P
pbrook 已提交
732
    int devfn;
733

P
pbrook 已提交
734 735
    while (bus && bus->bus_num != bus_num)
        bus = bus->next;
P
pbrook 已提交
736 737 738 739 740 741
    if (bus) {
        for(devfn = 0; devfn < 256; devfn++) {
            d = bus->devices[devfn];
            if (d)
                fn(d);
        }
B
bellard 已提交
742 743 744
    }
}

A
aliguori 已提交
745
void pci_info(Monitor *mon)
B
bellard 已提交
746
{
P
pbrook 已提交
747
    pci_for_each_device(0, pci_info_device);
B
bellard 已提交
748
}
749

750
PCIDevice *pci_create(const char *name, const char *devaddr)
751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767
{
    PCIBus *bus;
    int devfn;
    DeviceState *dev;

    bus = pci_get_bus_devfn(&devfn, devaddr);
    if (!bus) {
        fprintf(stderr, "Invalid PCI device address %s for device %s\n",
                devaddr, name);
        exit(1);
    }

    dev = qdev_create(&bus->qbus, name);
    qdev_set_prop_int(dev, "devfn", devfn);
    return (PCIDevice *)dev;
}

768 769 770 771 772 773 774 775 776 777 778 779
static const char * const pci_nic_models[] = {
    "ne2k_pci",
    "i82551",
    "i82557b",
    "i82559er",
    "rtl8139",
    "e1000",
    "pcnet",
    "virtio",
    NULL
};

P
Paul Brook 已提交
780 781 782 783 784 785 786 787
static const char * const pci_nic_names[] = {
    "ne2k_pci",
    "i82551",
    "i82557b",
    "i82559er",
    "rtl8139",
    "e1000",
    "pcnet",
P
Paul Brook 已提交
788
    "virtio-net-pci",
789 790 791
    NULL
};

792
/* Initialize a PCI NIC.  */
793 794
PCIDevice *pci_nic_init(NICInfo *nd, const char *default_model,
                        const char *default_devaddr)
795
{
796 797
    const char *devaddr = nd->devaddr ? nd->devaddr : default_devaddr;
    PCIDevice *pci_dev;
P
Paul Brook 已提交
798
    DeviceState *dev;
799 800 801 802
    int i;

    qemu_check_nic_model_list(nd, pci_nic_models, default_model);

P
Paul Brook 已提交
803
    for (i = 0; pci_nic_models[i]; i++) {
804
        if (strcmp(nd->model, pci_nic_models[i]) == 0) {
805 806
            pci_dev = pci_create(pci_nic_names[i], devaddr);
            dev = &pci_dev->qdev;
P
Paul Brook 已提交
807 808 809
            qdev_set_netdev(dev, nd);
            qdev_init(dev);
            nd->private = dev;
810
            return pci_dev;
811
        }
P
Paul Brook 已提交
812
    }
813 814

    return NULL;
815 816
}

P
pbrook 已提交
817 818 819 820 821
typedef struct {
    PCIDevice dev;
    PCIBus *bus;
} PCIBridge;

822
static void pci_bridge_write_config(PCIDevice *d,
P
pbrook 已提交
823 824 825 826 827
                             uint32_t address, uint32_t val, int len)
{
    PCIBridge *s = (PCIBridge *)d;

    pci_default_write_config(d, address, val, len);
828
    s->bus->bus_num = d->config[PCI_SECONDARY_BUS];
P
pbrook 已提交
829 830
}

831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850
PCIBus *pci_find_bus(int bus_num)
{
    PCIBus *bus = first_bus;

    while (bus && bus->bus_num != bus_num)
        bus = bus->next;

    return bus;
}

PCIDevice *pci_find_device(int bus_num, int slot, int function)
{
    PCIBus *bus = pci_find_bus(bus_num);

    if (!bus)
        return NULL;

    return bus->devices[PCI_DEVFN(slot, function)];
}

B
blueswir1 已提交
851
PCIBus *pci_bridge_init(PCIBus *bus, int devfn, uint16_t vid, uint16_t did,
P
pbrook 已提交
852 853 854
                        pci_map_irq_fn map_irq, const char *name)
{
    PCIBridge *s;
855
    s = (PCIBridge *)pci_register_device(bus, name, sizeof(PCIBridge),
P
pbrook 已提交
856
                                         devfn, NULL, pci_bridge_write_config);
B
blueswir1 已提交
857 858 859 860

    pci_config_set_vendor_id(s->dev.config, vid);
    pci_config_set_device_id(s->dev.config, did);

P
pbrook 已提交
861 862 863 864 865 866
    s->dev.config[0x04] = 0x06; // command = bus master, pci mem
    s->dev.config[0x05] = 0x00;
    s->dev.config[0x06] = 0xa0; // status = fast back-to-back, 66MHz, no error
    s->dev.config[0x07] = 0x00; // status = fast devsel
    s->dev.config[0x08] = 0x00; // revision
    s->dev.config[0x09] = 0x00; // programming i/f
867
    pci_config_set_class(s->dev.config, PCI_CLASS_BRIDGE_PCI);
P
pbrook 已提交
868
    s->dev.config[0x0D] = 0x10; // latency_timer
I
Isaku Yamahata 已提交
869 870
    s->dev.config[PCI_HEADER_TYPE] =
        PCI_HEADER_TYPE_MULTI_FUNCTION | PCI_HEADER_TYPE_BRIDGE; // header_type
P
pbrook 已提交
871 872 873 874 875
    s->dev.config[0x1E] = 0xa0; // secondary status

    s->bus = pci_register_secondary_bus(&s->dev, map_irq);
    return s->bus;
}
P
Paul Brook 已提交
876

P
Paul Brook 已提交
877 878 879 880 881 882
typedef struct {
    DeviceInfo qdev;
    pci_qdev_initfn init;
} PCIDeviceInfo;

static void pci_qdev_init(DeviceState *qdev, DeviceInfo *base)
P
Paul Brook 已提交
883 884
{
    PCIDevice *pci_dev = (PCIDevice *)qdev;
P
Paul Brook 已提交
885
    PCIDeviceInfo *info = container_of(base, PCIDeviceInfo, qdev);
P
Paul Brook 已提交
886 887 888
    PCIBus *bus;
    int devfn;

P
Paul Brook 已提交
889
    bus = FROM_QBUS(PCIBus, qdev_get_parent_bus(qdev));
P
Paul Brook 已提交
890 891 892 893
    devfn = qdev_get_prop_int(qdev, "devfn", -1);
    pci_dev = do_pci_register_device(pci_dev, bus, "FIXME", devfn,
                                     NULL, NULL);//FIXME:config_read, config_write);
    assert(pci_dev);
P
Paul Brook 已提交
894
    info->init(pci_dev);
P
Paul Brook 已提交
895 896 897 898
}

void pci_qdev_register(const char *name, int size, pci_qdev_initfn init)
{
P
Paul Brook 已提交
899 900 901
    PCIDeviceInfo *info;

    info = qemu_mallocz(sizeof(*info));
902 903
    info->qdev.name = qemu_strdup(name);
    info->qdev.size = size;
P
Paul Brook 已提交
904 905 906 907
    info->init = init;
    info->qdev.init = pci_qdev_init;
    info->qdev.bus_type = BUS_TYPE_PCI;

908
    qdev_register(&info->qdev);
P
Paul Brook 已提交
909 910 911 912 913 914
}

PCIDevice *pci_create_simple(PCIBus *bus, int devfn, const char *name)
{
    DeviceState *dev;

P
Paul Brook 已提交
915
    dev = qdev_create(&bus->qbus, name);
P
Paul Brook 已提交
916 917 918 919 920
    qdev_set_prop_int(dev, "devfn", devfn);
    qdev_init(dev);

    return (PCIDevice *)dev;
}
921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940 941 942 943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965

static int pci_find_space(PCIDevice *pdev, uint8_t size)
{
    int offset = PCI_CONFIG_HEADER_SIZE;
    int i;
    for (i = PCI_CONFIG_HEADER_SIZE; i < PCI_CONFIG_SPACE_SIZE; ++i)
        if (pdev->used[i])
            offset = i + 1;
        else if (i - offset + 1 == size)
            return offset;
    return 0;
}

static uint8_t pci_find_capability_list(PCIDevice *pdev, uint8_t cap_id,
                                        uint8_t *prev_p)
{
    uint8_t next, prev;

    if (!(pdev->config[PCI_STATUS] & PCI_STATUS_CAP_LIST))
        return 0;

    for (prev = PCI_CAPABILITY_LIST; (next = pdev->config[prev]);
         prev = next + PCI_CAP_LIST_NEXT)
        if (pdev->config[next + PCI_CAP_LIST_ID] == cap_id)
            break;

    if (prev_p)
        *prev_p = prev;
    return next;
}

/* Reserve space and add capability to the linked list in pci config space */
int pci_add_capability(PCIDevice *pdev, uint8_t cap_id, uint8_t size)
{
    uint8_t offset = pci_find_space(pdev, size);
    uint8_t *config = pdev->config + offset;
    if (!offset)
        return -ENOSPC;
    config[PCI_CAP_LIST_ID] = cap_id;
    config[PCI_CAP_LIST_NEXT] = pdev->config[PCI_CAPABILITY_LIST];
    pdev->config[PCI_CAPABILITY_LIST] = offset;
    pdev->config[PCI_STATUS] |= PCI_STATUS_CAP_LIST;
    memset(pdev->used + offset, 0xFF, size);
    /* Make capability read-only by default */
    memset(pdev->wmask + offset, 0, size);
966 967
    /* Check capability by default */
    memset(pdev->cmask + offset, 0xFF, size);
968 969 970 971 972 973 974 975 976 977 978 979
    return offset;
}

/* Unlink capability from the pci config space. */
void pci_del_capability(PCIDevice *pdev, uint8_t cap_id, uint8_t size)
{
    uint8_t prev, offset = pci_find_capability_list(pdev, cap_id, &prev);
    if (!offset)
        return;
    pdev->config[prev] = pdev->config[offset + PCI_CAP_LIST_NEXT];
    /* Make capability writeable again */
    memset(pdev->wmask + offset, 0xff, size);
980 981
    /* Clear cmask as device-specific registers can't be checked */
    memset(pdev->cmask + offset, 0, size);
982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997
    memset(pdev->used + offset, 0, size);

    if (!pdev->config[PCI_CAPABILITY_LIST])
        pdev->config[PCI_STATUS] &= ~PCI_STATUS_CAP_LIST;
}

/* Reserve space for capability at a known offset (to call after load). */
void pci_reserve_capability(PCIDevice *pdev, uint8_t offset, uint8_t size)
{
    memset(pdev->used + offset, 0xff, size);
}

uint8_t pci_find_capability(PCIDevice *pdev, uint8_t cap_id)
{
    return pci_find_capability_list(pdev, cap_id, NULL);
}