ahci-test.c 47.2 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 23 24 25 26 27
/*
 * AHCI test cases
 *
 * Copyright (c) 2014 John Snow <jsnow@redhat.com>
 *
 * 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.
 */

#include <stdint.h>
#include <string.h>
#include <stdio.h>
28
#include <getopt.h>
29 30 31
#include <glib.h>

#include "libqtest.h"
J
John Snow 已提交
32
#include "libqos/libqos-pc.h"
J
John Snow 已提交
33
#include "libqos/ahci.h"
34 35 36 37 38 39 40 41
#include "libqos/pci-pc.h"

#include "qemu-common.h"
#include "qemu/host-utils.h"

#include "hw/pci/pci_ids.h"
#include "hw/pci/pci_regs.h"

42 43
/* Test-specific defines -- in MiB */
#define TEST_IMAGE_SIZE_MB (200 * 1024)
44 45
#define TEST_IMAGE_SECTORS ((TEST_IMAGE_SIZE_MB / AHCI_SECTOR_SIZE)     \
                            * 1024 * 1024)
46 47 48

/*** Globals ***/
static char tmp_path[] = "/tmp/qtest.XXXXXX";
J
John Snow 已提交
49
static char debug_path[] = "/tmp/qtest-blkdebug.XXXXXX";
50 51
static bool ahci_pedantic;

52
/*** Function Declarations ***/
53
static void ahci_test_port_spec(AHCIQState *ahci, uint8_t port);
54 55
static void ahci_test_pci_spec(AHCIQState *ahci);
static void ahci_test_pci_caps(AHCIQState *ahci, uint16_t header,
56
                               uint8_t offset);
57 58 59
static void ahci_test_satacap(AHCIQState *ahci, uint8_t offset);
static void ahci_test_msicap(AHCIQState *ahci, uint8_t offset);
static void ahci_test_pmcap(AHCIQState *ahci, uint8_t offset);
60 61 62

/*** Utilities ***/

63 64 65 66 67 68 69 70 71 72 73
static void string_bswap16(uint16_t *s, size_t bytes)
{
    g_assert_cmphex((bytes & 1), ==, 0);
    bytes /= 2;

    while (bytes--) {
        *s = bswap16(*s);
        s++;
    }
}

74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99
static void generate_pattern(void *buffer, size_t len, size_t cycle_len)
{
    int i, j;
    unsigned char *tx = (unsigned char *)buffer;
    unsigned char p;
    size_t *sx;

    /* Write an indicative pattern that varies and is unique per-cycle */
    p = rand() % 256;
    for (i = j = 0; i < len; i++, j++) {
        tx[i] = p;
        if (j % cycle_len == 0) {
            p = rand() % 256;
        }
    }

    /* force uniqueness by writing an id per-cycle */
    for (i = 0; i < len / cycle_len; i++) {
        j = i * cycle_len;
        if (j + sizeof(*sx) <= len) {
            sx = (size_t *)&tx[j];
            *sx = i;
        }
    }
}

J
John Snow 已提交
100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165
/**
 * Verify that the transfer did not corrupt our state at all.
 */
static void verify_state(AHCIQState *ahci)
{
    int i, j;
    uint32_t ahci_fingerprint;
    uint64_t hba_base;
    uint64_t hba_stored;
    AHCICommandHeader cmd;

    ahci_fingerprint = qpci_config_readl(ahci->dev, PCI_VENDOR_ID);
    g_assert_cmphex(ahci_fingerprint, ==, ahci->fingerprint);

    /* If we haven't initialized, this is as much as can be validated. */
    if (!ahci->hba_base) {
        return;
    }

    hba_base = (uint64_t)qpci_config_readl(ahci->dev, PCI_BASE_ADDRESS_5);
    hba_stored = (uint64_t)(uintptr_t)ahci->hba_base;
    g_assert_cmphex(hba_base, ==, hba_stored);

    g_assert_cmphex(ahci_rreg(ahci, AHCI_CAP), ==, ahci->cap);
    g_assert_cmphex(ahci_rreg(ahci, AHCI_CAP2), ==, ahci->cap2);

    for (i = 0; i < 32; i++) {
        g_assert_cmphex(ahci_px_rreg(ahci, i, AHCI_PX_FB), ==,
                        ahci->port[i].fb);
        g_assert_cmphex(ahci_px_rreg(ahci, i, AHCI_PX_CLB), ==,
                        ahci->port[i].clb);
        for (j = 0; j < 32; j++) {
            ahci_get_command_header(ahci, i, j, &cmd);
            g_assert_cmphex(cmd.prdtl, ==, ahci->port[i].prdtl[j]);
            g_assert_cmphex(cmd.ctba, ==, ahci->port[i].ctba[j]);
        }
    }
}

static void ahci_migrate(AHCIQState *from, AHCIQState *to, const char *uri)
{
    QOSState *tmp = to->parent;
    QPCIDevice *dev = to->dev;
    if (uri == NULL) {
        uri = "tcp:127.0.0.1:1234";
    }

    /* context will be 'to' after completion. */
    migrate(from->parent, to->parent, uri);

    /* We'd like for the AHCIState objects to still point
     * to information specific to its specific parent
     * instance, but otherwise just inherit the new data. */
    memcpy(to, from, sizeof(AHCIQState));
    to->parent = tmp;
    to->dev = dev;

    tmp = from->parent;
    dev = from->dev;
    memset(from, 0x00, sizeof(AHCIQState));
    from->parent = tmp;
    from->dev = dev;

    verify_state(to);
}

166 167 168
/*** Test Setup & Teardown ***/

/**
J
John Snow 已提交
169
 * Start a Q35 machine and bookmark a handle to the AHCI device.
170
 */
171
static AHCIQState *ahci_vboot(const char *cli, va_list ap)
172
{
J
John Snow 已提交
173
    AHCIQState *s;
174

J
John Snow 已提交
175
    s = g_malloc0(sizeof(AHCIQState));
176
    s->parent = qtest_pc_vboot(cli, ap);
J
John Snow 已提交
177
    alloc_set_flags(s->parent->alloc, ALLOC_LEAK_ASSERT);
178

J
John Snow 已提交
179
    /* Verify that we have an AHCI device present. */
180
    s->dev = get_ahci_device(&s->fingerprint);
181

J
John Snow 已提交
182
    return s;
183 184
}

185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208
/**
 * Start a Q35 machine and bookmark a handle to the AHCI device.
 */
static AHCIQState *ahci_boot(const char *cli, ...)
{
    AHCIQState *s;
    va_list ap;

    if (cli) {
        va_start(ap, cli);
        s = ahci_vboot(cli, ap);
        va_end(ap);
    } else {
        cli = "-drive if=none,id=drive0,file=%s,cache=writeback,serial=%s"
            ",format=qcow2"
            " -M q35 "
            "-device ide-hd,drive=drive0 "
            "-global ide-hd.ver=%s";
        s = ahci_boot(cli, tmp_path, "testdisk", "version");
    }

    return s;
}

209 210 211
/**
 * Clean up the PCI device, then terminate the QEMU instance.
 */
J
John Snow 已提交
212
static void ahci_shutdown(AHCIQState *ahci)
213
{
J
John Snow 已提交
214
    QOSState *qs = ahci->parent;
J
John Snow 已提交
215 216

    set_context(qs);
J
John Snow 已提交
217
    ahci_clean_mem(ahci);
J
John Snow 已提交
218 219 220
    free_ahci_device(ahci->dev);
    g_free(ahci);
    qtest_shutdown(qs);
221 222
}

223 224 225 226
/**
 * Boot and fully enable the HBA device.
 * @see ahci_boot, ahci_pci_enable and ahci_hba_enable.
 */
227
static AHCIQState *ahci_boot_and_enable(const char *cli, ...)
228 229
{
    AHCIQState *ahci;
230 231 232 233 234 235 236 237 238
    va_list ap;

    if (cli) {
        va_start(ap, cli);
        ahci = ahci_vboot(cli, ap);
        va_end(ap);
    } else {
        ahci = ahci_boot(NULL);
    }
239 240 241 242 243 244 245

    ahci_pci_enable(ahci);
    ahci_hba_enable(ahci);

    return ahci;
}

246 247 248 249 250
/*** Specification Adherence Tests ***/

/**
 * Implementation for test_pci_spec. Ensures PCI configuration space is sane.
 */
251
static void ahci_test_pci_spec(AHCIQState *ahci)
252 253 254 255 256 257
{
    uint8_t datab;
    uint16_t data;
    uint32_t datal;

    /* Most of these bits should start cleared until we turn them on. */
258
    data = qpci_config_readw(ahci->dev, PCI_COMMAND);
259 260 261 262 263 264 265 266 267 268 269
    ASSERT_BIT_CLEAR(data, PCI_COMMAND_MEMORY);
    ASSERT_BIT_CLEAR(data, PCI_COMMAND_MASTER);
    ASSERT_BIT_CLEAR(data, PCI_COMMAND_SPECIAL);     /* Reserved */
    ASSERT_BIT_CLEAR(data, PCI_COMMAND_VGA_PALETTE); /* Reserved */
    ASSERT_BIT_CLEAR(data, PCI_COMMAND_PARITY);
    ASSERT_BIT_CLEAR(data, PCI_COMMAND_WAIT);        /* Reserved */
    ASSERT_BIT_CLEAR(data, PCI_COMMAND_SERR);
    ASSERT_BIT_CLEAR(data, PCI_COMMAND_FAST_BACK);
    ASSERT_BIT_CLEAR(data, PCI_COMMAND_INTX_DISABLE);
    ASSERT_BIT_CLEAR(data, 0xF800);                  /* Reserved */

270
    data = qpci_config_readw(ahci->dev, PCI_STATUS);
271 272 273 274 275 276 277 278 279 280 281 282
    ASSERT_BIT_CLEAR(data, 0x01 | 0x02 | 0x04);     /* Reserved */
    ASSERT_BIT_CLEAR(data, PCI_STATUS_INTERRUPT);
    ASSERT_BIT_SET(data, PCI_STATUS_CAP_LIST);      /* must be set */
    ASSERT_BIT_CLEAR(data, PCI_STATUS_UDF);         /* Reserved */
    ASSERT_BIT_CLEAR(data, PCI_STATUS_PARITY);
    ASSERT_BIT_CLEAR(data, PCI_STATUS_SIG_TARGET_ABORT);
    ASSERT_BIT_CLEAR(data, PCI_STATUS_REC_TARGET_ABORT);
    ASSERT_BIT_CLEAR(data, PCI_STATUS_REC_MASTER_ABORT);
    ASSERT_BIT_CLEAR(data, PCI_STATUS_SIG_SYSTEM_ERROR);
    ASSERT_BIT_CLEAR(data, PCI_STATUS_DETECTED_PARITY);

    /* RID occupies the low byte, CCs occupy the high three. */
283
    datal = qpci_config_readl(ahci->dev, PCI_CLASS_REVISION);
284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305
    if (ahci_pedantic) {
        /* AHCI 1.3 specifies that at-boot, the RID should reset to 0x00,
         * Though in practice this is likely seldom true. */
        ASSERT_BIT_CLEAR(datal, 0xFF);
    }

    /* BCC *must* equal 0x01. */
    g_assert_cmphex(PCI_BCC(datal), ==, 0x01);
    if (PCI_SCC(datal) == 0x01) {
        /* IDE */
        ASSERT_BIT_SET(0x80000000, datal);
        ASSERT_BIT_CLEAR(0x60000000, datal);
    } else if (PCI_SCC(datal) == 0x04) {
        /* RAID */
        g_assert_cmphex(PCI_PI(datal), ==, 0);
    } else if (PCI_SCC(datal) == 0x06) {
        /* AHCI */
        g_assert_cmphex(PCI_PI(datal), ==, 0x01);
    } else {
        g_assert_not_reached();
    }

306
    datab = qpci_config_readb(ahci->dev, PCI_CACHE_LINE_SIZE);
307 308
    g_assert_cmphex(datab, ==, 0);

309
    datab = qpci_config_readb(ahci->dev, PCI_LATENCY_TIMER);
310 311 312
    g_assert_cmphex(datab, ==, 0);

    /* Only the bottom 7 bits must be off. */
313
    datab = qpci_config_readb(ahci->dev, PCI_HEADER_TYPE);
314 315 316
    ASSERT_BIT_CLEAR(datab, 0x7F);

    /* BIST is optional, but the low 7 bits must always start off regardless. */
317
    datab = qpci_config_readb(ahci->dev, PCI_BIST);
318 319 320
    ASSERT_BIT_CLEAR(datab, 0x7F);

    /* BARS 0-4 do not have a boot spec, but ABAR/BAR5 must be clean. */
321
    datal = qpci_config_readl(ahci->dev, PCI_BASE_ADDRESS_5);
322 323
    g_assert_cmphex(datal, ==, 0);

324 325
    qpci_config_writel(ahci->dev, PCI_BASE_ADDRESS_5, 0xFFFFFFFF);
    datal = qpci_config_readl(ahci->dev, PCI_BASE_ADDRESS_5);
326 327 328 329 330
    /* ABAR must be 32-bit, memory mapped, non-prefetchable and
     * must be >= 512 bytes. To that end, bits 0-8 must be off. */
    ASSERT_BIT_CLEAR(datal, 0xFF);

    /* Capability list MUST be present, */
331
    datal = qpci_config_readl(ahci->dev, PCI_CAPABILITY_LIST);
332 333 334 335 336
    /* But these bits are reserved. */
    ASSERT_BIT_CLEAR(datal, ~0xFF);
    g_assert_cmphex(datal, !=, 0);

    /* Check specification adherence for capability extenstions. */
337
    data = qpci_config_readw(ahci->dev, datal);
338

339
    switch (ahci->fingerprint) {
340 341 342 343 344 345 346 347 348 349 350 351
    case AHCI_INTEL_ICH9:
        /* Intel ICH9 Family Datasheet 14.1.19 p.550 */
        g_assert_cmphex((data & 0xFF), ==, PCI_CAP_ID_MSI);
        break;
    default:
        /* AHCI 1.3, Section 2.1.14 -- CAP must point to PMCAP. */
        g_assert_cmphex((data & 0xFF), ==, PCI_CAP_ID_PM);
    }

    ahci_test_pci_caps(ahci, data, (uint8_t)datal);

    /* Reserved. */
352
    datal = qpci_config_readl(ahci->dev, PCI_CAPABILITY_LIST + 4);
353 354 355
    g_assert_cmphex(datal, ==, 0);

    /* IPIN might vary, but ILINE must be off. */
356
    datab = qpci_config_readb(ahci->dev, PCI_INTERRUPT_LINE);
357 358 359 360 361 362
    g_assert_cmphex(datab, ==, 0);
}

/**
 * Test PCI capabilities for AHCI specification adherence.
 */
363
static void ahci_test_pci_caps(AHCIQState *ahci, uint16_t header,
364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386
                               uint8_t offset)
{
    uint8_t cid = header & 0xFF;
    uint8_t next = header >> 8;

    g_test_message("CID: %02x; next: %02x", cid, next);

    switch (cid) {
    case PCI_CAP_ID_PM:
        ahci_test_pmcap(ahci, offset);
        break;
    case PCI_CAP_ID_MSI:
        ahci_test_msicap(ahci, offset);
        break;
    case PCI_CAP_ID_SATA:
        ahci_test_satacap(ahci, offset);
        break;

    default:
        g_test_message("Unknown CAP 0x%02x", cid);
    }

    if (next) {
387
        ahci_test_pci_caps(ahci, qpci_config_readw(ahci->dev, next), next);
388 389 390 391 392 393
    }
}

/**
 * Test SATA PCI capabilitity for AHCI specification adherence.
 */
394
static void ahci_test_satacap(AHCIQState *ahci, uint8_t offset)
395 396 397 398 399 400 401
{
    uint16_t dataw;
    uint32_t datal;

    g_test_message("Verifying SATACAP");

    /* Assert that the SATACAP version is 1.0, And reserved bits are empty. */
402
    dataw = qpci_config_readw(ahci->dev, offset + 2);
403 404 405
    g_assert_cmphex(dataw, ==, 0x10);

    /* Grab the SATACR1 register. */
406
    datal = qpci_config_readw(ahci->dev, offset + 4);
407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428

    switch (datal & 0x0F) {
    case 0x04: /* BAR0 */
    case 0x05: /* BAR1 */
    case 0x06:
    case 0x07:
    case 0x08:
    case 0x09: /* BAR5 */
    case 0x0F: /* Immediately following SATACR1 in PCI config space. */
        break;
    default:
        /* Invalid BARLOC for the Index Data Pair. */
        g_assert_not_reached();
    }

    /* Reserved. */
    g_assert_cmphex((datal >> 24), ==, 0x00);
}

/**
 * Test MSI PCI capability for AHCI specification adherence.
 */
429
static void ahci_test_msicap(AHCIQState *ahci, uint8_t offset)
430 431 432 433 434 435
{
    uint16_t dataw;
    uint32_t datal;

    g_test_message("Verifying MSICAP");

436
    dataw = qpci_config_readw(ahci->dev, offset + PCI_MSI_FLAGS);
437 438 439 440
    ASSERT_BIT_CLEAR(dataw, PCI_MSI_FLAGS_ENABLE);
    ASSERT_BIT_CLEAR(dataw, PCI_MSI_FLAGS_QSIZE);
    ASSERT_BIT_CLEAR(dataw, PCI_MSI_FLAGS_RESERVED);

441
    datal = qpci_config_readl(ahci->dev, offset + PCI_MSI_ADDRESS_LO);
442 443 444 445
    g_assert_cmphex(datal, ==, 0);

    if (dataw & PCI_MSI_FLAGS_64BIT) {
        g_test_message("MSICAP is 64bit");
446
        datal = qpci_config_readl(ahci->dev, offset + PCI_MSI_ADDRESS_HI);
447
        g_assert_cmphex(datal, ==, 0);
448
        dataw = qpci_config_readw(ahci->dev, offset + PCI_MSI_DATA_64);
449 450 451
        g_assert_cmphex(dataw, ==, 0);
    } else {
        g_test_message("MSICAP is 32bit");
452
        dataw = qpci_config_readw(ahci->dev, offset + PCI_MSI_DATA_32);
453 454 455 456 457 458 459
        g_assert_cmphex(dataw, ==, 0);
    }
}

/**
 * Test Power Management PCI capability for AHCI specification adherence.
 */
460
static void ahci_test_pmcap(AHCIQState *ahci, uint8_t offset)
461 462 463 464 465
{
    uint16_t dataw;

    g_test_message("Verifying PMCAP");

466
    dataw = qpci_config_readw(ahci->dev, offset + PCI_PM_PMC);
467 468 469 470 471
    ASSERT_BIT_CLEAR(dataw, PCI_PM_CAP_PME_CLOCK);
    ASSERT_BIT_CLEAR(dataw, PCI_PM_CAP_RESERVED);
    ASSERT_BIT_CLEAR(dataw, PCI_PM_CAP_D1);
    ASSERT_BIT_CLEAR(dataw, PCI_PM_CAP_D2);

472
    dataw = qpci_config_readw(ahci->dev, offset + PCI_PM_CTRL);
473 474 475 476 477 478
    ASSERT_BIT_CLEAR(dataw, PCI_PM_CTRL_STATE_MASK);
    ASSERT_BIT_CLEAR(dataw, PCI_PM_CTRL_RESERVED);
    ASSERT_BIT_CLEAR(dataw, PCI_PM_CTRL_DATA_SEL_MASK);
    ASSERT_BIT_CLEAR(dataw, PCI_PM_CTRL_DATA_SCALE_MASK);
}

479
static void ahci_test_hba_spec(AHCIQState *ahci)
480 481
{
    unsigned i;
482
    uint32_t reg;
483 484 485 486
    uint32_t ports;
    uint8_t nports_impl;
    uint8_t maxports;

487
    g_assert(ahci != NULL);
488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509

    /*
     * Note that the AHCI spec does expect the BIOS to set up a few things:
     * CAP.SSS    - Support for staggered spin-up            (t/f)
     * CAP.SMPS   - Support for mechanical presence switches (t/f)
     * PI         - Ports Implemented                        (1-32)
     * PxCMD.HPCP - Hot Plug Capable Port
     * PxCMD.MPSP - Mechanical Presence Switch Present
     * PxCMD.CPD  - Cold Presence Detection support
     *
     * Additional items are touched if CAP.SSS is on, see AHCI 10.1.1 p.97:
     * Foreach Port Implemented:
     * -PxCMD.ST, PxCMD.CR, PxCMD.FRE, PxCMD.FR, PxSCTL.DET are 0
     * -PxCLB/U and PxFB/U are set to valid regions in memory
     * -PxSUD is set to 1.
     * -PxSSTS.DET is polled for presence; if detected, we continue:
     * -PxSERR is cleared with 1's.
     * -If PxTFD.STS.BSY, PxTFD.STS.DRQ, and PxTFD.STS.ERR are all zero,
     *  the device is ready.
     */

    /* 1 CAP - Capabilities Register */
510
    ahci->cap = ahci_rreg(ahci, AHCI_CAP);
511
    ASSERT_BIT_CLEAR(ahci->cap, AHCI_CAP_RESERVED);
512 513

    /* 2 GHC - Global Host Control */
514
    reg = ahci_rreg(ahci, AHCI_GHC);
515 516 517
    ASSERT_BIT_CLEAR(reg, AHCI_GHC_HR);
    ASSERT_BIT_CLEAR(reg, AHCI_GHC_IE);
    ASSERT_BIT_CLEAR(reg, AHCI_GHC_MRSM);
518
    if (BITSET(ahci->cap, AHCI_CAP_SAM)) {
519 520 521 522 523 524 525 526
        g_test_message("Supports AHCI-Only Mode: GHC_AE is Read-Only.");
        ASSERT_BIT_SET(reg, AHCI_GHC_AE);
    } else {
        g_test_message("Supports AHCI/Legacy mix.");
        ASSERT_BIT_CLEAR(reg, AHCI_GHC_AE);
    }

    /* 3 IS - Interrupt Status */
527
    reg = ahci_rreg(ahci, AHCI_IS);
528 529 530
    g_assert_cmphex(reg, ==, 0);

    /* 4 PI - Ports Implemented */
531
    ports = ahci_rreg(ahci, AHCI_PI);
532 533 534 535
    /* Ports Implemented must be non-zero. */
    g_assert_cmphex(ports, !=, 0);
    /* Ports Implemented must be <= Number of Ports. */
    nports_impl = ctpopl(ports);
536
    g_assert_cmpuint(((AHCI_CAP_NP & ahci->cap) + 1), >=, nports_impl);
537 538 539 540

    /* Ports must be within the proper range. Given a mapping of SIZE,
     * 256 bytes are used for global HBA control, and the rest is used
     * for ports data, at 0x80 bytes each. */
541 542
    g_assert_cmphex(ahci->barsize, >, 0);
    maxports = (ahci->barsize - HBA_DATA_REGION_SIZE) / HBA_PORT_DATA_SIZE;
543 544 545 546
    /* e.g, 30 ports for 4K of memory. (4096 - 256) / 128 = 30 */
    g_assert_cmphex((reg >> maxports), ==, 0);

    /* 5 AHCI Version */
547
    reg = ahci_rreg(ahci, AHCI_VS);
548 549 550 551 552 553 554 555 556 557 558 559
    switch (reg) {
    case AHCI_VERSION_0_95:
    case AHCI_VERSION_1_0:
    case AHCI_VERSION_1_1:
    case AHCI_VERSION_1_2:
    case AHCI_VERSION_1_3:
        break;
    default:
        g_assert_not_reached();
    }

    /* 6 Command Completion Coalescing Control: depends on CAP.CCCS. */
560
    reg = ahci_rreg(ahci, AHCI_CCCCTL);
561
    if (BITSET(ahci->cap, AHCI_CAP_CCCS)) {
562 563 564 565 566 567 568 569 570
        ASSERT_BIT_CLEAR(reg, AHCI_CCCCTL_EN);
        ASSERT_BIT_CLEAR(reg, AHCI_CCCCTL_RESERVED);
        ASSERT_BIT_SET(reg, AHCI_CCCCTL_CC);
        ASSERT_BIT_SET(reg, AHCI_CCCCTL_TV);
    } else {
        g_assert_cmphex(reg, ==, 0);
    }

    /* 7 CCC_PORTS */
571
    reg = ahci_rreg(ahci, AHCI_CCCPORTS);
572 573 574 575
    /* Must be zeroes initially regardless of CAP.CCCS */
    g_assert_cmphex(reg, ==, 0);

    /* 8 EM_LOC */
576
    reg = ahci_rreg(ahci, AHCI_EMLOC);
577
    if (BITCLR(ahci->cap, AHCI_CAP_EMS)) {
578 579 580 581
        g_assert_cmphex(reg, ==, 0);
    }

    /* 9 EM_CTL */
582
    reg = ahci_rreg(ahci, AHCI_EMCTL);
583
    if (BITSET(ahci->cap, AHCI_CAP_EMS)) {
584 585 586 587 588 589 590 591 592
        ASSERT_BIT_CLEAR(reg, AHCI_EMCTL_STSMR);
        ASSERT_BIT_CLEAR(reg, AHCI_EMCTL_CTLTM);
        ASSERT_BIT_CLEAR(reg, AHCI_EMCTL_CTLRST);
        ASSERT_BIT_CLEAR(reg, AHCI_EMCTL_RESERVED);
    } else {
        g_assert_cmphex(reg, ==, 0);
    }

    /* 10 CAP2 -- Capabilities Extended */
593
    ahci->cap2 = ahci_rreg(ahci, AHCI_CAP2);
594
    ASSERT_BIT_CLEAR(ahci->cap2, AHCI_CAP2_RESERVED);
595 596

    /* 11 BOHC -- Bios/OS Handoff Control */
597
    reg = ahci_rreg(ahci, AHCI_BOHC);
598 599 600 601 602
    g_assert_cmphex(reg, ==, 0);

    /* 12 -- 23: Reserved */
    g_test_message("Verifying HBA reserved area is empty.");
    for (i = AHCI_RESERVED; i < AHCI_NVMHCI; ++i) {
603
        reg = ahci_rreg(ahci, i);
604 605 606 607
        g_assert_cmphex(reg, ==, 0);
    }

    /* 24 -- 39: NVMHCI */
608
    if (BITCLR(ahci->cap2, AHCI_CAP2_NVMP)) {
609 610
        g_test_message("Verifying HBA/NVMHCI area is empty.");
        for (i = AHCI_NVMHCI; i < AHCI_VENDOR; ++i) {
611
            reg = ahci_rreg(ahci, i);
612 613 614 615 616 617 618
            g_assert_cmphex(reg, ==, 0);
        }
    }

    /* 40 -- 63: Vendor */
    g_test_message("Verifying HBA/Vendor area is empty.");
    for (i = AHCI_VENDOR; i < AHCI_PORTS; ++i) {
619
        reg = ahci_rreg(ahci, i);
620 621 622 623 624 625 626
        g_assert_cmphex(reg, ==, 0);
    }

    /* 64 -- XX: Port Space */
    for (i = 0; ports || (i < maxports); ports >>= 1, ++i) {
        if (BITSET(ports, 0x1)) {
            g_test_message("Testing port %u for spec", i);
627
            ahci_test_port_spec(ahci, i);
628 629 630 631 632 633 634 635
        } else {
            uint16_t j;
            uint16_t low = AHCI_PORTS + (32 * i);
            uint16_t high = AHCI_PORTS + (32 * (i + 1));
            g_test_message("Asserting unimplemented port %u "
                           "(reg [%u-%u]) is empty.",
                           i, low, high - 1);
            for (j = low; j < high; ++j) {
636
                reg = ahci_rreg(ahci, j);
637 638 639 640 641 642 643 644 645
                g_assert_cmphex(reg, ==, 0);
            }
        }
    }
}

/**
 * Test the memory space for one port for specification adherence.
 */
646
static void ahci_test_port_spec(AHCIQState *ahci, uint8_t port)
647 648 649 650 651
{
    uint32_t reg;
    unsigned i;

    /* (0) CLB */
652
    reg = ahci_px_rreg(ahci, port, AHCI_PX_CLB);
653 654 655
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CLB_RESERVED);

    /* (1) CLBU */
656
    if (BITCLR(ahci->cap, AHCI_CAP_S64A)) {
657
        reg = ahci_px_rreg(ahci, port, AHCI_PX_CLBU);
658 659 660 661
        g_assert_cmphex(reg, ==, 0);
    }

    /* (2) FB */
662
    reg = ahci_px_rreg(ahci, port, AHCI_PX_FB);
663 664 665
    ASSERT_BIT_CLEAR(reg, AHCI_PX_FB_RESERVED);

    /* (3) FBU */
666
    if (BITCLR(ahci->cap, AHCI_CAP_S64A)) {
667
        reg = ahci_px_rreg(ahci, port, AHCI_PX_FBU);
668 669 670 671
        g_assert_cmphex(reg, ==, 0);
    }

    /* (4) IS */
672
    reg = ahci_px_rreg(ahci, port, AHCI_PX_IS);
673 674 675
    g_assert_cmphex(reg, ==, 0);

    /* (5) IE */
676
    reg = ahci_px_rreg(ahci, port, AHCI_PX_IE);
677 678 679
    g_assert_cmphex(reg, ==, 0);

    /* (6) CMD */
680
    reg = ahci_px_rreg(ahci, port, AHCI_PX_CMD);
681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_FRE);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_RESERVED);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_CCS);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_FR);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_CR);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_PMA); /* And RW only if CAP.SPM */
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_APSTE); /* RW only if CAP2.APST */
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_ATAPI);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_DLAE);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_ALPE);  /* RW only if CAP.SALP */
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_ASP);   /* RW only if CAP.SALP */
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_ICC);
    /* If CPDetect support does not exist, CPState must be off. */
    if (BITCLR(reg, AHCI_PX_CMD_CPD)) {
        ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_CPS);
    }
    /* If MPSPresence is not set, MPSState must be off. */
    if (BITCLR(reg, AHCI_PX_CMD_MPSP)) {
        ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_MPSS);
    }
    /* If we do not support MPS, MPSS and MPSP must be off. */
702
    if (BITCLR(ahci->cap, AHCI_CAP_SMPS)) {
703 704 705 706
        ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_MPSS);
        ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_MPSP);
    }
    /* If, via CPD or MPSP we detect a drive, HPCP must be on. */
707
    if (BITANY(reg, AHCI_PX_CMD_CPD | AHCI_PX_CMD_MPSP)) {
708 709 710 711 712
        ASSERT_BIT_SET(reg, AHCI_PX_CMD_HPCP);
    }
    /* HPCP and ESP cannot both be active. */
    g_assert(!BITSET(reg, AHCI_PX_CMD_HPCP | AHCI_PX_CMD_ESP));
    /* If CAP.FBSS is not set, FBSCP must not be set. */
713
    if (BITCLR(ahci->cap, AHCI_CAP_FBSS)) {
714 715 716 717
        ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_FBSCP);
    }

    /* (7) RESERVED */
718
    reg = ahci_px_rreg(ahci, port, AHCI_PX_RES1);
719 720 721
    g_assert_cmphex(reg, ==, 0);

    /* (8) TFD */
722
    reg = ahci_px_rreg(ahci, port, AHCI_PX_TFD);
723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739
    /* At boot, prior to an FIS being received, the TFD register should be 0x7F,
     * which breaks down as follows, as seen in AHCI 1.3 sec 3.3.8, p. 27. */
    ASSERT_BIT_SET(reg, AHCI_PX_TFD_STS_ERR);
    ASSERT_BIT_SET(reg, AHCI_PX_TFD_STS_CS1);
    ASSERT_BIT_SET(reg, AHCI_PX_TFD_STS_DRQ);
    ASSERT_BIT_SET(reg, AHCI_PX_TFD_STS_CS2);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_TFD_STS_BSY);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_TFD_ERR);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_TFD_RESERVED);

    /* (9) SIG */
    /* Though AHCI specifies the boot value should be 0xFFFFFFFF,
     * Even when GHC.ST is zero, the AHCI HBA may receive the initial
     * D2H register FIS and update the signature asynchronously,
     * so we cannot expect a value here. AHCI 1.3, sec 3.3.9, pp 27-28 */

    /* (10) SSTS / SCR0: SStatus */
740
    reg = ahci_px_rreg(ahci, port, AHCI_PX_SSTS);
741 742 743 744 745
    ASSERT_BIT_CLEAR(reg, AHCI_PX_SSTS_RESERVED);
    /* Even though the register should be 0 at boot, it is asynchronous and
     * prone to change, so we cannot test any well known value. */

    /* (11) SCTL / SCR2: SControl */
746
    reg = ahci_px_rreg(ahci, port, AHCI_PX_SCTL);
747 748 749
    g_assert_cmphex(reg, ==, 0);

    /* (12) SERR / SCR1: SError */
750
    reg = ahci_px_rreg(ahci, port, AHCI_PX_SERR);
751 752 753
    g_assert_cmphex(reg, ==, 0);

    /* (13) SACT / SCR3: SActive */
754
    reg = ahci_px_rreg(ahci, port, AHCI_PX_SACT);
755 756 757
    g_assert_cmphex(reg, ==, 0);

    /* (14) CI */
758
    reg = ahci_px_rreg(ahci, port, AHCI_PX_CI);
759 760 761
    g_assert_cmphex(reg, ==, 0);

    /* (15) SNTF */
762
    reg = ahci_px_rreg(ahci, port, AHCI_PX_SNTF);
763 764 765
    g_assert_cmphex(reg, ==, 0);

    /* (16) FBS */
766
    reg = ahci_px_rreg(ahci, port, AHCI_PX_FBS);
767 768 769 770 771 772
    ASSERT_BIT_CLEAR(reg, AHCI_PX_FBS_EN);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_FBS_DEC);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_FBS_SDE);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_FBS_DEV);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_FBS_DWE);
    ASSERT_BIT_CLEAR(reg, AHCI_PX_FBS_RESERVED);
773
    if (BITSET(ahci->cap, AHCI_CAP_FBSS)) {
774 775 776 777 778 779
        /* if Port-Multiplier FIS-based switching avail, ADO must >= 2 */
        g_assert((reg & AHCI_PX_FBS_ADO) >> ctzl(AHCI_PX_FBS_ADO) >= 2);
    }

    /* [17 -- 27] RESERVED */
    for (i = AHCI_PX_RES2; i < AHCI_PX_VS; ++i) {
780
        reg = ahci_px_rreg(ahci, port, i);
781 782 783 784 785
        g_assert_cmphex(reg, ==, 0);
    }

    /* [28 -- 31] Vendor-Specific */
    for (i = AHCI_PX_VS; i < 32; ++i) {
786
        reg = ahci_px_rreg(ahci, port, i);
787 788 789 790 791 792
        if (reg) {
            g_test_message("INFO: Vendor register %u non-empty", i);
        }
    }
}

793 794 795 796
/**
 * Utilizing an initialized AHCI HBA, issue an IDENTIFY command to the first
 * device we see, then read and check the response.
 */
797
static void ahci_test_identify(AHCIQState *ahci)
798 799
{
    uint16_t buff[256];
J
John Snow 已提交
800
    unsigned px;
801
    int rc;
802
    uint16_t sect_size;
J
John Snow 已提交
803
    const size_t buffsize = 512;
804 805 806

    g_assert(ahci != NULL);

J
John Snow 已提交
807 808 809 810 811 812
    /**
     * This serves as a bit of a tutorial on AHCI device programming:
     *
     * (1) Create a data buffer for the IDENTIFY response to be sent to
     * (2) Create a Command Table buffer, where we will store the
     *     command and PRDT (Physical Region Descriptor Table)
813
     * (3) Construct an FIS host-to-device command structure, and write it to
J
John Snow 已提交
814 815 816 817 818 819 820 821 822 823 824 825
     *     the top of the Command Table buffer.
     * (4) Create one or more Physical Region Descriptors (PRDs) that describe
     *     a location in memory where data may be stored/retrieved.
     * (5) Write these PRDTs to the bottom (offset 0x80) of the Command Table.
     * (6) Each AHCI port has up to 32 command slots. Each slot contains a
     *     header that points to a Command Table buffer. Pick an unused slot
     *     and update it to point to the Command Table we have built.
     * (7) Now: Command #n points to our Command Table, and our Command Table
     *     contains the FIS (that describes our command) and the PRDTL, which
     *     describes our buffer.
     * (8) We inform the HBA via PxCI (Command Issue) that the command in slot
     *     #n is ready for processing.
826 827 828
     */

    /* Pick the first implemented and running port */
J
John Snow 已提交
829 830
    px = ahci_port_select(ahci);
    g_test_message("Selected port %u for test", px);
831

832
    /* Clear out the FIS Receive area and any pending interrupts. */
J
John Snow 已提交
833
    ahci_port_clear(ahci, px);
834

J
John Snow 已提交
835
    /* "Read" 512 bytes using CMD_IDENTIFY into the host buffer. */
836
    ahci_io(ahci, px, CMD_IDENTIFY, &buff, buffsize, 0);
837 838 839 840 841 842 843 844 845 846 847 848 849

    /* Check serial number/version in the buffer */
    /* NB: IDENTIFY strings are packed in 16bit little endian chunks.
     * Since we copy byte-for-byte in ahci-test, on both LE and BE, we need to
     * unchunk this data. By contrast, ide-test copies 2 bytes at a time, and
     * as a consequence, only needs to unchunk the data on LE machines. */
    string_bswap16(&buff[10], 20);
    rc = memcmp(&buff[10], "testdisk            ", 20);
    g_assert_cmphex(rc, ==, 0);

    string_bswap16(&buff[23], 8);
    rc = memcmp(&buff[23], "version ", 8);
    g_assert_cmphex(rc, ==, 0);
850 851

    sect_size = le16_to_cpu(*((uint16_t *)(&buff[5])));
852
    g_assert_cmphex(sect_size, ==, AHCI_SECTOR_SIZE);
853 854
}

J
John Snow 已提交
855
static void ahci_test_io_rw_simple(AHCIQState *ahci, unsigned bufsize,
856 857
                                   uint64_t sector, uint8_t read_cmd,
                                   uint8_t write_cmd)
858 859 860 861 862 863 864 865 866 867 868 869 870 871 872 873 874
{
    uint64_t ptr;
    uint8_t port;
    unsigned char *tx = g_malloc(bufsize);
    unsigned char *rx = g_malloc0(bufsize);

    g_assert(ahci != NULL);

    /* Pick the first running port and clear it. */
    port = ahci_port_select(ahci);
    ahci_port_clear(ahci, port);

    /*** Create pattern and transfer to guest ***/
    /* Data buffer in the guest */
    ptr = ahci_alloc(ahci, bufsize);
    g_assert(ptr);

J
John Snow 已提交
875
    /* Write some indicative pattern to our buffer. */
876
    generate_pattern(tx, bufsize, AHCI_SECTOR_SIZE);
877
    bufwrite(ptr, tx, bufsize);
878 879

    /* Write this buffer to disk, then read it back to the DMA buffer. */
880
    ahci_guest_io(ahci, port, write_cmd, ptr, bufsize, sector);
881
    qmemset(ptr, 0x00, bufsize);
882
    ahci_guest_io(ahci, port, read_cmd, ptr, bufsize, sector);
883 884

    /*** Read back the Data ***/
885
    bufread(ptr, rx, bufsize);
886 887 888 889 890 891 892
    g_assert_cmphex(memcmp(tx, rx, bufsize), ==, 0);

    ahci_free(ahci, ptr);
    g_free(tx);
    g_free(rx);
}

J
John Snow 已提交
893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915
static void ahci_test_nondata(AHCIQState *ahci, uint8_t ide_cmd)
{
    uint8_t px;
    AHCICommand *cmd;

    /* Sanitize */
    px = ahci_port_select(ahci);
    ahci_port_clear(ahci, px);

    /* Issue Command */
    cmd = ahci_command_create(ide_cmd);
    ahci_command_commit(ahci, cmd, px);
    ahci_command_issue(ahci, cmd);
    ahci_command_verify(ahci, cmd);
    ahci_command_free(cmd);
}

static void ahci_test_flush(AHCIQState *ahci)
{
    ahci_test_nondata(ahci, CMD_FLUSH_CACHE);
}


916 917 918 919 920 921 922 923 924
/******************************************************************************/
/* Test Interfaces                                                            */
/******************************************************************************/

/**
 * Basic sanity test to boot a machine, find an AHCI device, and shutdown.
 */
static void test_sanity(void)
{
J
John Snow 已提交
925
    AHCIQState *ahci;
926
    ahci = ahci_boot(NULL);
927 928 929
    ahci_shutdown(ahci);
}

930 931 932 933 934 935
/**
 * Ensure that the PCI configuration space for the AHCI device is in-line with
 * the AHCI 1.3 specification for initial values.
 */
static void test_pci_spec(void)
{
J
John Snow 已提交
936
    AHCIQState *ahci;
937
    ahci = ahci_boot(NULL);
938
    ahci_test_pci_spec(ahci);
939 940 941
    ahci_shutdown(ahci);
}

942 943 944 945 946 947
/**
 * Engage the PCI AHCI device and sanity check the response.
 * Perform additional PCI config space bringup for the HBA.
 */
static void test_pci_enable(void)
{
J
John Snow 已提交
948
    AHCIQState *ahci;
949
    ahci = ahci_boot(NULL);
950
    ahci_pci_enable(ahci);
951 952 953
    ahci_shutdown(ahci);
}

954 955 956 957 958 959
/**
 * Investigate the memory mapped regions of the HBA,
 * and test them for AHCI specification adherence.
 */
static void test_hba_spec(void)
{
J
John Snow 已提交
960
    AHCIQState *ahci;
961

962
    ahci = ahci_boot(NULL);
963 964
    ahci_pci_enable(ahci);
    ahci_test_hba_spec(ahci);
965 966 967
    ahci_shutdown(ahci);
}

968 969 970 971 972 973
/**
 * Engage the HBA functionality of the AHCI PCI device,
 * and bring it into a functional idle state.
 */
static void test_hba_enable(void)
{
J
John Snow 已提交
974
    AHCIQState *ahci;
975

976
    ahci = ahci_boot(NULL);
977 978
    ahci_pci_enable(ahci);
    ahci_hba_enable(ahci);
979 980 981
    ahci_shutdown(ahci);
}

982 983 984 985 986 987
/**
 * Bring up the device and issue an IDENTIFY command.
 * Inspect the state of the HBA device and the data returned.
 */
static void test_identify(void)
{
J
John Snow 已提交
988
    AHCIQState *ahci;
989

990
    ahci = ahci_boot_and_enable(NULL);
991
    ahci_test_identify(ahci);
992 993 994
    ahci_shutdown(ahci);
}

J
John Snow 已提交
995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008 1009 1010
/**
 * Fragmented DMA test: Perform a standard 4K DMA read/write
 * test, but make sure the physical regions are fragmented to
 * be very small, each just 32 bytes, to see how AHCI performs
 * with chunks defined to be much less than a sector.
 */
static void test_dma_fragmented(void)
{
    AHCIQState *ahci;
    AHCICommand *cmd;
    uint8_t px;
    size_t bufsize = 4096;
    unsigned char *tx = g_malloc(bufsize);
    unsigned char *rx = g_malloc0(bufsize);
    uint64_t ptr;

1011
    ahci = ahci_boot_and_enable(NULL);
J
John Snow 已提交
1012 1013 1014 1015
    px = ahci_port_select(ahci);
    ahci_port_clear(ahci, px);

    /* create pattern */
1016
    generate_pattern(tx, bufsize, AHCI_SECTOR_SIZE);
J
John Snow 已提交
1017 1018 1019 1020

    /* Create a DMA buffer in guest memory, and write our pattern to it. */
    ptr = guest_alloc(ahci->parent->alloc, bufsize);
    g_assert(ptr);
1021
    bufwrite(ptr, tx, bufsize);
J
John Snow 已提交
1022 1023 1024 1025 1026 1027 1028 1029 1030 1031 1032 1033 1034 1035 1036 1037

    cmd = ahci_command_create(CMD_WRITE_DMA);
    ahci_command_adjust(cmd, 0, ptr, bufsize, 32);
    ahci_command_commit(ahci, cmd, px);
    ahci_command_issue(ahci, cmd);
    ahci_command_verify(ahci, cmd);
    g_free(cmd);

    cmd = ahci_command_create(CMD_READ_DMA);
    ahci_command_adjust(cmd, 0, ptr, bufsize, 32);
    ahci_command_commit(ahci, cmd, px);
    ahci_command_issue(ahci, cmd);
    ahci_command_verify(ahci, cmd);
    g_free(cmd);

    /* Read back the guest's receive buffer into local memory */
1038
    bufread(ptr, rx, bufsize);
J
John Snow 已提交
1039 1040 1041 1042 1043 1044 1045 1046 1047 1048
    guest_free(ahci->parent->alloc, ptr);

    g_assert_cmphex(memcmp(tx, rx, bufsize), ==, 0);

    ahci_shutdown(ahci);

    g_free(rx);
    g_free(tx);
}

J
John Snow 已提交
1049 1050 1051 1052
static void test_flush(void)
{
    AHCIQState *ahci;

1053
    ahci = ahci_boot_and_enable(NULL);
J
John Snow 已提交
1054 1055 1056 1057
    ahci_test_flush(ahci);
    ahci_shutdown(ahci);
}

J
John Snow 已提交
1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073
static void test_flush_retry(void)
{
    AHCIQState *ahci;
    AHCICommand *cmd;
    uint8_t port;
    const char *s;

    prepare_blkdebug_script(debug_path, "flush_to_disk");
    ahci = ahci_boot_and_enable("-drive file=blkdebug:%s:%s,if=none,id=drive0,"
                                "format=qcow2,cache=writeback,"
                                "rerror=stop,werror=stop "
                                "-M q35 "
                                "-device ide-hd,drive=drive0 ",
                                debug_path,
                                tmp_path);

J
John Snow 已提交
1074
    /* Issue Flush Command and wait for error */
J
John Snow 已提交
1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092
    port = ahci_port_select(ahci);
    ahci_port_clear(ahci, port);
    cmd = ahci_command_create(CMD_FLUSH_CACHE);
    ahci_command_commit(ahci, cmd, port);
    ahci_command_issue_async(ahci, cmd);
    qmp_eventwait("STOP");

    /* Complete the command */
    s = "{'execute':'cont' }";
    qmp_async(s);
    qmp_eventwait("RESUME");
    ahci_command_wait(ahci, cmd);
    ahci_command_verify(ahci, cmd);

    ahci_command_free(cmd);
    ahci_shutdown(ahci);
}

J
John Snow 已提交
1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112
/**
 * Basic sanity test to boot a machine, find an AHCI device, and shutdown.
 */
static void test_migrate_sanity(void)
{
    AHCIQState *src, *dst;
    const char *uri = "tcp:127.0.0.1:1234";

    src = ahci_boot("-m 1024 -M q35 "
                    "-hda %s ", tmp_path);
    dst = ahci_boot("-m 1024 -M q35 "
                    "-hda %s "
                    "-incoming %s", tmp_path, uri);

    ahci_migrate(src, dst, uri);

    ahci_shutdown(src);
    ahci_shutdown(dst);
}

J
John Snow 已提交
1113 1114 1115 1116 1117 1118 1119 1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156
/**
 * DMA Migration test: Write a pattern, migrate, then read.
 */
static void test_migrate_dma(void)
{
    AHCIQState *src, *dst;
    uint8_t px;
    size_t bufsize = 4096;
    unsigned char *tx = g_malloc(bufsize);
    unsigned char *rx = g_malloc0(bufsize);
    unsigned i;
    const char *uri = "tcp:127.0.0.1:1234";

    src = ahci_boot_and_enable("-m 1024 -M q35 "
                               "-hda %s ", tmp_path);
    dst = ahci_boot("-m 1024 -M q35 "
                    "-hda %s "
                    "-incoming %s", tmp_path, uri);

    set_context(src->parent);

    /* initialize */
    px = ahci_port_select(src);
    ahci_port_clear(src, px);

    /* create pattern */
    for (i = 0; i < bufsize; i++) {
        tx[i] = (bufsize - i);
    }

    /* Write, migrate, then read. */
    ahci_io(src, px, CMD_WRITE_DMA, tx, bufsize, 0);
    ahci_migrate(src, dst, uri);
    ahci_io(dst, px, CMD_READ_DMA, rx, bufsize, 0);

    /* Verify pattern */
    g_assert_cmphex(memcmp(tx, rx, bufsize), ==, 0);

    ahci_shutdown(src);
    ahci_shutdown(dst);
    g_free(rx);
    g_free(tx);
}

J
John Snow 已提交
1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203 1204 1205 1206 1207 1208 1209 1210 1211 1212 1213 1214 1215
/**
 * DMA Error Test
 *
 * Simulate an error on first write, Try to write a pattern,
 * Confirm the VM has stopped, resume the VM, verify command
 * has completed, then read back the data and verify.
 */
static void test_halted_dma(void)
{
    AHCIQState *ahci;
    uint8_t port;
    size_t bufsize = 4096;
    unsigned char *tx = g_malloc(bufsize);
    unsigned char *rx = g_malloc0(bufsize);
    unsigned i;
    uint64_t ptr;
    AHCICommand *cmd;

    prepare_blkdebug_script(debug_path, "write_aio");

    ahci = ahci_boot_and_enable("-drive file=blkdebug:%s:%s,if=none,id=drive0,"
                                "format=qcow2,cache=writeback,"
                                "rerror=stop,werror=stop "
                                "-M q35 "
                                "-device ide-hd,drive=drive0 ",
                                debug_path,
                                tmp_path);

    /* Initialize and prepare */
    port = ahci_port_select(ahci);
    ahci_port_clear(ahci, port);

    for (i = 0; i < bufsize; i++) {
        tx[i] = (bufsize - i);
    }

    /* create DMA source buffer and write pattern */
    ptr = ahci_alloc(ahci, bufsize);
    g_assert(ptr);
    memwrite(ptr, tx, bufsize);

    /* Attempt to write (and fail) */
    cmd = ahci_guest_io_halt(ahci, port, CMD_WRITE_DMA,
                             ptr, bufsize, 0);

    /* Attempt to resume the command */
    ahci_guest_io_resume(ahci, cmd);
    ahci_free(ahci, ptr);

    /* Read back and verify */
    ahci_io(ahci, port, CMD_READ_DMA, rx, bufsize, 0);
    g_assert_cmphex(memcmp(tx, rx, bufsize), ==, 0);

    /* Cleanup and go home */
    ahci_shutdown(ahci);
    g_free(rx);
    g_free(tx);
}

1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260 1261 1262 1263 1264 1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287
/**
 * DMA Error Migration Test
 *
 * Simulate an error on first write, Try to write a pattern,
 * Confirm the VM has stopped, migrate, resume the VM,
 * verify command has completed, then read back the data and verify.
 */
static void test_migrate_halted_dma(void)
{
    AHCIQState *src, *dst;
    uint8_t port;
    size_t bufsize = 4096;
    unsigned char *tx = g_malloc(bufsize);
    unsigned char *rx = g_malloc0(bufsize);
    unsigned i;
    uint64_t ptr;
    AHCICommand *cmd;
    const char *uri = "tcp:127.0.0.1:1234";

    prepare_blkdebug_script(debug_path, "write_aio");

    src = ahci_boot_and_enable("-drive file=blkdebug:%s:%s,if=none,id=drive0,"
                               "format=qcow2,cache=writeback,"
                               "rerror=stop,werror=stop "
                               "-M q35 "
                               "-device ide-hd,drive=drive0 ",
                               debug_path,
                               tmp_path);

    dst = ahci_boot("-drive file=%s,if=none,id=drive0,"
                    "format=qcow2,cache=writeback,"
                    "rerror=stop,werror=stop "
                    "-M q35 "
                    "-device ide-hd,drive=drive0 "
                    "-incoming %s",
                    tmp_path, uri);

    set_context(src->parent);

    /* Initialize and prepare */
    port = ahci_port_select(src);
    ahci_port_clear(src, port);

    for (i = 0; i < bufsize; i++) {
        tx[i] = (bufsize - i);
    }

    /* create DMA source buffer and write pattern */
    ptr = ahci_alloc(src, bufsize);
    g_assert(ptr);
    memwrite(ptr, tx, bufsize);

    /* Write, trigger the VM to stop, migrate, then resume. */
    cmd = ahci_guest_io_halt(src, port, CMD_WRITE_DMA,
                             ptr, bufsize, 0);
    ahci_migrate(src, dst, uri);
    ahci_guest_io_resume(dst, cmd);
    ahci_free(dst, ptr);

    /* Read back */
    ahci_io(dst, port, CMD_READ_DMA, rx, bufsize, 0);

    /* Verify TX and RX are identical */
    g_assert_cmphex(memcmp(tx, rx, bufsize), ==, 0);

    /* Cleanup and go home. */
    ahci_shutdown(src);
    ahci_shutdown(dst);
    g_free(rx);
    g_free(tx);
}

J
John Snow 已提交
1288 1289 1290 1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308 1309 1310 1311 1312 1313 1314 1315 1316 1317 1318 1319 1320 1321 1322 1323 1324 1325 1326 1327 1328 1329 1330 1331 1332 1333 1334 1335 1336
/**
 * Migration test: Try to flush, migrate, then resume.
 */
static void test_flush_migrate(void)
{
    AHCIQState *src, *dst;
    AHCICommand *cmd;
    uint8_t px;
    const char *s;
    const char *uri = "tcp:127.0.0.1:1234";

    prepare_blkdebug_script(debug_path, "flush_to_disk");

    src = ahci_boot_and_enable("-drive file=blkdebug:%s:%s,if=none,id=drive0,"
                               "cache=writeback,rerror=stop,werror=stop "
                               "-M q35 "
                               "-device ide-hd,drive=drive0 ",
                               debug_path, tmp_path);
    dst = ahci_boot("-drive file=%s,if=none,id=drive0,"
                    "cache=writeback,rerror=stop,werror=stop "
                    "-M q35 "
                    "-device ide-hd,drive=drive0 "
                    "-incoming %s", tmp_path, uri);

    set_context(src->parent);

    /* Issue Flush Command */
    px = ahci_port_select(src);
    ahci_port_clear(src, px);
    cmd = ahci_command_create(CMD_FLUSH_CACHE);
    ahci_command_commit(src, cmd, px);
    ahci_command_issue_async(src, cmd);
    qmp_eventwait("STOP");

    /* Migrate over */
    ahci_migrate(src, dst, uri);

    /* Complete the command */
    s = "{'execute':'cont' }";
    qmp_async(s);
    qmp_eventwait("RESUME");
    ahci_command_wait(dst, cmd);
    ahci_command_verify(dst, cmd);

    ahci_command_free(cmd);
    ahci_shutdown(src);
    ahci_shutdown(dst);
}

J
John Snow 已提交
1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373 1374 1375 1376
/******************************************************************************/
/* AHCI I/O Test Matrix Definitions                                           */

enum BuffLen {
    LEN_BEGIN = 0,
    LEN_SIMPLE = LEN_BEGIN,
    LEN_DOUBLE,
    LEN_LONG,
    LEN_SHORT,
    NUM_LENGTHS
};

static const char *buff_len_str[NUM_LENGTHS] = { "simple", "double",
                                                 "long", "short" };

enum AddrMode {
    ADDR_MODE_BEGIN = 0,
    ADDR_MODE_LBA28 = ADDR_MODE_BEGIN,
    ADDR_MODE_LBA48,
    NUM_ADDR_MODES
};

static const char *addr_mode_str[NUM_ADDR_MODES] = { "lba28", "lba48" };

enum IOMode {
    MODE_BEGIN = 0,
    MODE_PIO = MODE_BEGIN,
    MODE_DMA,
    NUM_MODES
};

static const char *io_mode_str[NUM_MODES] = { "pio", "dma" };

enum IOOps {
    IO_BEGIN = 0,
    IO_READ = IO_BEGIN,
    IO_WRITE,
    NUM_IO_OPS
};

1377 1378 1379 1380 1381 1382 1383 1384 1385 1386
enum OffsetType {
    OFFSET_BEGIN = 0,
    OFFSET_ZERO = OFFSET_BEGIN,
    OFFSET_LOW,
    OFFSET_HIGH,
    NUM_OFFSETS
};

static const char *offset_str[NUM_OFFSETS] = { "zero", "low", "high" };

J
John Snow 已提交
1387 1388 1389 1390
typedef struct AHCIIOTestOptions {
    enum BuffLen length;
    enum AddrMode address_type;
    enum IOMode io_type;
1391
    enum OffsetType offset;
J
John Snow 已提交
1392 1393
} AHCIIOTestOptions;

1394 1395 1396 1397 1398 1399 1400 1401 1402 1403 1404 1405 1406 1407 1408 1409 1410 1411 1412 1413 1414 1415
static uint64_t offset_sector(enum OffsetType ofst,
                              enum AddrMode addr_type,
                              uint64_t buffsize)
{
    uint64_t ceil;
    uint64_t nsectors;

    switch (ofst) {
    case OFFSET_ZERO:
        return 0;
    case OFFSET_LOW:
        return 1;
    case OFFSET_HIGH:
        ceil = (addr_type == ADDR_MODE_LBA28) ? 0xfffffff : 0xffffffffffff;
        ceil = MIN(ceil, TEST_IMAGE_SECTORS - 1);
        nsectors = buffsize / AHCI_SECTOR_SIZE;
        return ceil - nsectors + 1;
    default:
        g_assert_not_reached();
    }
}

1416
/**
J
John Snow 已提交
1417
 * Table of possible I/O ATA commands given a set of enumerations.
1418
 */
J
John Snow 已提交
1419 1420 1421 1422 1423 1424 1425 1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442
static const uint8_t io_cmds[NUM_MODES][NUM_ADDR_MODES][NUM_IO_OPS] = {
    [MODE_PIO] = {
        [ADDR_MODE_LBA28] = {
            [IO_READ] = CMD_READ_PIO,
            [IO_WRITE] = CMD_WRITE_PIO },
        [ADDR_MODE_LBA48] = {
            [IO_READ] = CMD_READ_PIO_EXT,
            [IO_WRITE] = CMD_WRITE_PIO_EXT }
    },
    [MODE_DMA] = {
        [ADDR_MODE_LBA28] = {
            [IO_READ] = CMD_READ_DMA,
            [IO_WRITE] = CMD_WRITE_DMA },
        [ADDR_MODE_LBA48] = {
            [IO_READ] = CMD_READ_DMA_EXT,
            [IO_WRITE] = CMD_WRITE_DMA_EXT }
    }
};

/**
 * Test a Read/Write pattern using various commands, addressing modes,
 * transfer modes, and buffer sizes.
 */
static void test_io_rw_interface(enum AddrMode lba48, enum IOMode dma,
1443
                                 unsigned bufsize, uint64_t sector)
1444 1445 1446
{
    AHCIQState *ahci;

1447
    ahci = ahci_boot_and_enable(NULL);
1448
    ahci_test_io_rw_simple(ahci, bufsize, sector,
J
John Snow 已提交
1449 1450
                           io_cmds[dma][lba48][IO_READ],
                           io_cmds[dma][lba48][IO_WRITE]);
1451 1452 1453
    ahci_shutdown(ahci);
}

J
John Snow 已提交
1454 1455 1456 1457
/**
 * Demultiplex the test data and invoke the actual test routine.
 */
static void test_io_interface(gconstpointer opaque)
J
John Snow 已提交
1458
{
J
John Snow 已提交
1459 1460
    AHCIIOTestOptions *opts = (AHCIIOTestOptions *)opaque;
    unsigned bufsize;
1461
    uint64_t sector;
J
John Snow 已提交
1462

J
John Snow 已提交
1463 1464 1465 1466 1467 1468 1469 1470 1471 1472 1473 1474 1475 1476 1477 1478
    switch (opts->length) {
    case LEN_SIMPLE:
        bufsize = 4096;
        break;
    case LEN_DOUBLE:
        bufsize = 8192;
        break;
    case LEN_LONG:
        bufsize = 4096 * 64;
        break;
    case LEN_SHORT:
        bufsize = 512;
        break;
    default:
        g_assert_not_reached();
    }
J
John Snow 已提交
1479

1480 1481
    sector = offset_sector(opts->offset, opts->address_type, bufsize);
    test_io_rw_interface(opts->address_type, opts->io_type, bufsize, sector);
J
John Snow 已提交
1482 1483
    g_free(opts);
    return;
J
John Snow 已提交
1484 1485
}

J
John Snow 已提交
1486
static void create_ahci_io_test(enum IOMode type, enum AddrMode addr,
1487
                                enum BuffLen len, enum OffsetType offset)
J
John Snow 已提交
1488
{
J
John Snow 已提交
1489 1490 1491 1492 1493 1494 1495
    static const char *arch;
    char *name;
    AHCIIOTestOptions *opts = g_malloc(sizeof(AHCIIOTestOptions));

    opts->length = len;
    opts->address_type = addr;
    opts->io_type = type;
1496
    opts->offset = offset;
J
John Snow 已提交
1497 1498 1499 1500 1501

    if (!arch) {
        arch = qtest_get_arch();
    }

1502
    name = g_strdup_printf("/%s/ahci/io/%s/%s/%s/%s", arch,
J
John Snow 已提交
1503 1504
                           io_mode_str[type],
                           addr_mode_str[addr],
1505 1506
                           buff_len_str[len],
                           offset_str[offset]);
J
John Snow 已提交
1507 1508 1509

    g_test_add_data_func(name, opts, test_io_interface);
    g_free(name);
J
John Snow 已提交
1510 1511
}

1512 1513 1514 1515 1516 1517
/******************************************************************************/

int main(int argc, char **argv)
{
    const char *arch;
    int ret;
J
John Snow 已提交
1518
    int fd;
1519
    int c;
1520
    int i, j, k, m;
1521 1522 1523 1524 1525

    static struct option long_options[] = {
        {"pedantic", no_argument, 0, 'p' },
        {0, 0, 0, 0},
    };
1526 1527 1528 1529

    /* Should be first to utilize g_test functionality, So we can see errors. */
    g_test_init(&argc, &argv, NULL);

1530 1531 1532 1533 1534 1535 1536 1537 1538 1539 1540 1541 1542 1543 1544 1545 1546
    while (1) {
        c = getopt_long(argc, argv, "", long_options, NULL);
        if (c == -1) {
            break;
        }
        switch (c) {
        case -1:
            break;
        case 'p':
            ahci_pedantic = 1;
            break;
        default:
            fprintf(stderr, "Unrecognized ahci_test option.\n");
            g_assert_not_reached();
        }
    }

1547 1548 1549 1550 1551 1552 1553
    /* Check architecture */
    arch = qtest_get_arch();
    if (strcmp(arch, "i386") && strcmp(arch, "x86_64")) {
        g_test_message("Skipping test for non-x86");
        return 0;
    }

1554 1555 1556
    /* Create a temporary qcow2 image */
    close(mkstemp(tmp_path));
    mkqcow2(tmp_path, TEST_IMAGE_SIZE_MB);
1557

J
John Snow 已提交
1558 1559 1560 1561 1562
    /* Create temporary blkdebug instructions */
    fd = mkstemp(debug_path);
    g_assert(fd >= 0);
    close(fd);

1563 1564
    /* Run the tests */
    qtest_add_func("/ahci/sanity",     test_sanity);
1565
    qtest_add_func("/ahci/pci_spec",   test_pci_spec);
1566
    qtest_add_func("/ahci/pci_enable", test_pci_enable);
1567
    qtest_add_func("/ahci/hba_spec",   test_hba_spec);
1568
    qtest_add_func("/ahci/hba_enable", test_hba_enable);
1569
    qtest_add_func("/ahci/identify",   test_identify);
J
John Snow 已提交
1570 1571 1572 1573

    for (i = MODE_BEGIN; i < NUM_MODES; i++) {
        for (j = ADDR_MODE_BEGIN; j < NUM_ADDR_MODES; j++) {
            for (k = LEN_BEGIN; k < NUM_LENGTHS; k++) {
1574 1575 1576
                for (m = OFFSET_BEGIN; m < NUM_OFFSETS; m++) {
                    create_ahci_io_test(i, j, k, m);
                }
J
John Snow 已提交
1577 1578 1579
            }
        }
    }
1580

J
John Snow 已提交
1581 1582
    qtest_add_func("/ahci/io/dma/lba28/fragmented", test_dma_fragmented);

J
John Snow 已提交
1583
    qtest_add_func("/ahci/flush/simple", test_flush);
J
John Snow 已提交
1584
    qtest_add_func("/ahci/flush/retry", test_flush_retry);
J
John Snow 已提交
1585
    qtest_add_func("/ahci/flush/migrate", test_flush_migrate);
J
John Snow 已提交
1586

J
John Snow 已提交
1587
    qtest_add_func("/ahci/migrate/sanity", test_migrate_sanity);
1588
    qtest_add_func("/ahci/migrate/dma/simple", test_migrate_dma);
J
John Snow 已提交
1589
    qtest_add_func("/ahci/io/dma/lba28/retry", test_halted_dma);
1590
    qtest_add_func("/ahci/migrate/dma/halted", test_migrate_halted_dma);
J
John Snow 已提交
1591

1592 1593 1594 1595
    ret = g_test_run();

    /* Cleanup */
    unlink(tmp_path);
J
John Snow 已提交
1596
    unlink(debug_path);
1597 1598 1599

    return ret;
}