ahci-test.c 30.1 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 42 43 44 45 46
#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"

/* Test-specific defines. */
#define TEST_IMAGE_SIZE    (64 * 1024 * 1024)

/*** Globals ***/
static char tmp_path[] = "/tmp/qtest.XXXXXX";
47 48
static bool ahci_pedantic;

49
/*** Function Declarations ***/
50
static void ahci_test_port_spec(AHCIQState *ahci, uint8_t port);
51 52
static void ahci_test_pci_spec(AHCIQState *ahci);
static void ahci_test_pci_caps(AHCIQState *ahci, uint16_t header,
53
                               uint8_t offset);
54 55 56
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);
57 58 59

/*** Utilities ***/

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

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

71 72 73
/*** Test Setup & Teardown ***/

/**
J
John Snow 已提交
74
 * Start a Q35 machine and bookmark a handle to the AHCI device.
75
 */
J
John Snow 已提交
76
static AHCIQState *ahci_boot(void)
77
{
J
John Snow 已提交
78 79
    AHCIQState *s;
    const char *cli;
80

J
John Snow 已提交
81
    s = g_malloc0(sizeof(AHCIQState));
82

J
John Snow 已提交
83 84 85 86 87
    cli = "-drive if=none,id=drive0,file=%s,cache=writeback,serial=%s"
        ",format=raw"
        " -M q35 "
        "-device ide-hd,drive=drive0 "
        "-global ide-hd.ver=%s";
J
John Snow 已提交
88
    s->parent = qtest_pc_boot(cli, tmp_path, "testdisk", "version");
89

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

J
John Snow 已提交
93
    return s;
94 95 96 97 98
}

/**
 * Clean up the PCI device, then terminate the QEMU instance.
 */
J
John Snow 已提交
99
static void ahci_shutdown(AHCIQState *ahci)
100
{
J
John Snow 已提交
101 102 103 104
    QOSState *qs = ahci->parent;
    free_ahci_device(ahci->dev);
    g_free(ahci);
    qtest_shutdown(qs);
105 106
}

107 108 109 110 111
/*** Specification Adherence Tests ***/

/**
 * Implementation for test_pci_spec. Ensures PCI configuration space is sane.
 */
112
static void ahci_test_pci_spec(AHCIQState *ahci)
113 114 115 116 117 118
{
    uint8_t datab;
    uint16_t data;
    uint32_t datal;

    /* Most of these bits should start cleared until we turn them on. */
119
    data = qpci_config_readw(ahci->dev, PCI_COMMAND);
120 121 122 123 124 125 126 127 128 129 130
    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 */

131
    data = qpci_config_readw(ahci->dev, PCI_STATUS);
132 133 134 135 136 137 138 139 140 141 142 143
    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. */
144
    datal = qpci_config_readl(ahci->dev, PCI_CLASS_REVISION);
145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166
    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();
    }

167
    datab = qpci_config_readb(ahci->dev, PCI_CACHE_LINE_SIZE);
168 169
    g_assert_cmphex(datab, ==, 0);

170
    datab = qpci_config_readb(ahci->dev, PCI_LATENCY_TIMER);
171 172 173
    g_assert_cmphex(datab, ==, 0);

    /* Only the bottom 7 bits must be off. */
174
    datab = qpci_config_readb(ahci->dev, PCI_HEADER_TYPE);
175 176 177
    ASSERT_BIT_CLEAR(datab, 0x7F);

    /* BIST is optional, but the low 7 bits must always start off regardless. */
178
    datab = qpci_config_readb(ahci->dev, PCI_BIST);
179 180 181
    ASSERT_BIT_CLEAR(datab, 0x7F);

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

185 186
    qpci_config_writel(ahci->dev, PCI_BASE_ADDRESS_5, 0xFFFFFFFF);
    datal = qpci_config_readl(ahci->dev, PCI_BASE_ADDRESS_5);
187 188 189 190 191
    /* 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, */
192
    datal = qpci_config_readl(ahci->dev, PCI_CAPABILITY_LIST);
193 194 195 196 197
    /* But these bits are reserved. */
    ASSERT_BIT_CLEAR(datal, ~0xFF);
    g_assert_cmphex(datal, !=, 0);

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

200
    switch (ahci->fingerprint) {
201 202 203 204 205 206 207 208 209 210 211 212
    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. */
213
    datal = qpci_config_readl(ahci->dev, PCI_CAPABILITY_LIST + 4);
214 215 216
    g_assert_cmphex(datal, ==, 0);

    /* IPIN might vary, but ILINE must be off. */
217
    datab = qpci_config_readb(ahci->dev, PCI_INTERRUPT_LINE);
218 219 220 221 222 223
    g_assert_cmphex(datab, ==, 0);
}

/**
 * Test PCI capabilities for AHCI specification adherence.
 */
224
static void ahci_test_pci_caps(AHCIQState *ahci, uint16_t header,
225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247
                               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) {
248
        ahci_test_pci_caps(ahci, qpci_config_readw(ahci->dev, next), next);
249 250 251 252 253 254
    }
}

/**
 * Test SATA PCI capabilitity for AHCI specification adherence.
 */
255
static void ahci_test_satacap(AHCIQState *ahci, uint8_t offset)
256 257 258 259 260 261 262
{
    uint16_t dataw;
    uint32_t datal;

    g_test_message("Verifying SATACAP");

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

    /* Grab the SATACR1 register. */
267
    datal = qpci_config_readw(ahci->dev, offset + 4);
268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289

    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.
 */
290
static void ahci_test_msicap(AHCIQState *ahci, uint8_t offset)
291 292 293 294 295 296
{
    uint16_t dataw;
    uint32_t datal;

    g_test_message("Verifying MSICAP");

297
    dataw = qpci_config_readw(ahci->dev, offset + PCI_MSI_FLAGS);
298 299 300 301
    ASSERT_BIT_CLEAR(dataw, PCI_MSI_FLAGS_ENABLE);
    ASSERT_BIT_CLEAR(dataw, PCI_MSI_FLAGS_QSIZE);
    ASSERT_BIT_CLEAR(dataw, PCI_MSI_FLAGS_RESERVED);

302
    datal = qpci_config_readl(ahci->dev, offset + PCI_MSI_ADDRESS_LO);
303 304 305 306
    g_assert_cmphex(datal, ==, 0);

    if (dataw & PCI_MSI_FLAGS_64BIT) {
        g_test_message("MSICAP is 64bit");
307
        datal = qpci_config_readl(ahci->dev, offset + PCI_MSI_ADDRESS_HI);
308
        g_assert_cmphex(datal, ==, 0);
309
        dataw = qpci_config_readw(ahci->dev, offset + PCI_MSI_DATA_64);
310 311 312
        g_assert_cmphex(dataw, ==, 0);
    } else {
        g_test_message("MSICAP is 32bit");
313
        dataw = qpci_config_readw(ahci->dev, offset + PCI_MSI_DATA_32);
314 315 316 317 318 319 320
        g_assert_cmphex(dataw, ==, 0);
    }
}

/**
 * Test Power Management PCI capability for AHCI specification adherence.
 */
321
static void ahci_test_pmcap(AHCIQState *ahci, uint8_t offset)
322 323 324 325 326
{
    uint16_t dataw;

    g_test_message("Verifying PMCAP");

327
    dataw = qpci_config_readw(ahci->dev, offset + PCI_PM_PMC);
328 329 330 331 332
    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);

333
    dataw = qpci_config_readw(ahci->dev, offset + PCI_PM_CTRL);
334 335 336 337 338 339
    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);
}

340
static void ahci_test_hba_spec(AHCIQState *ahci)
341 342
{
    unsigned i;
343
    uint32_t reg;
344 345 346 347
    uint32_t ports;
    uint8_t nports_impl;
    uint8_t maxports;

348
    g_assert(ahci != NULL);
349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370

    /*
     * 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 */
371
    ahci->cap = ahci_rreg(ahci, AHCI_CAP);
372
    ASSERT_BIT_CLEAR(ahci->cap, AHCI_CAP_RESERVED);
373 374

    /* 2 GHC - Global Host Control */
375
    reg = ahci_rreg(ahci, AHCI_GHC);
376 377 378
    ASSERT_BIT_CLEAR(reg, AHCI_GHC_HR);
    ASSERT_BIT_CLEAR(reg, AHCI_GHC_IE);
    ASSERT_BIT_CLEAR(reg, AHCI_GHC_MRSM);
379
    if (BITSET(ahci->cap, AHCI_CAP_SAM)) {
380 381 382 383 384 385 386 387
        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 */
388
    reg = ahci_rreg(ahci, AHCI_IS);
389 390 391
    g_assert_cmphex(reg, ==, 0);

    /* 4 PI - Ports Implemented */
392
    ports = ahci_rreg(ahci, AHCI_PI);
393 394 395 396
    /* Ports Implemented must be non-zero. */
    g_assert_cmphex(ports, !=, 0);
    /* Ports Implemented must be <= Number of Ports. */
    nports_impl = ctpopl(ports);
397
    g_assert_cmpuint(((AHCI_CAP_NP & ahci->cap) + 1), >=, nports_impl);
398 399 400 401

    /* 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. */
402 403
    g_assert_cmphex(ahci->barsize, >, 0);
    maxports = (ahci->barsize - HBA_DATA_REGION_SIZE) / HBA_PORT_DATA_SIZE;
404 405 406 407
    /* e.g, 30 ports for 4K of memory. (4096 - 256) / 128 = 30 */
    g_assert_cmphex((reg >> maxports), ==, 0);

    /* 5 AHCI Version */
408
    reg = ahci_rreg(ahci, AHCI_VS);
409 410 411 412 413 414 415 416 417 418 419 420
    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. */
421
    reg = ahci_rreg(ahci, AHCI_CCCCTL);
422
    if (BITSET(ahci->cap, AHCI_CAP_CCCS)) {
423 424 425 426 427 428 429 430 431
        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 */
432
    reg = ahci_rreg(ahci, AHCI_CCCPORTS);
433 434 435 436
    /* Must be zeroes initially regardless of CAP.CCCS */
    g_assert_cmphex(reg, ==, 0);

    /* 8 EM_LOC */
437
    reg = ahci_rreg(ahci, AHCI_EMLOC);
438
    if (BITCLR(ahci->cap, AHCI_CAP_EMS)) {
439 440 441 442
        g_assert_cmphex(reg, ==, 0);
    }

    /* 9 EM_CTL */
443
    reg = ahci_rreg(ahci, AHCI_EMCTL);
444
    if (BITSET(ahci->cap, AHCI_CAP_EMS)) {
445 446 447 448 449 450 451 452 453
        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 */
454
    ahci->cap2 = ahci_rreg(ahci, AHCI_CAP2);
455
    ASSERT_BIT_CLEAR(ahci->cap2, AHCI_CAP2_RESERVED);
456 457

    /* 11 BOHC -- Bios/OS Handoff Control */
458
    reg = ahci_rreg(ahci, AHCI_BOHC);
459 460 461 462 463
    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) {
464
        reg = ahci_rreg(ahci, i);
465 466 467 468
        g_assert_cmphex(reg, ==, 0);
    }

    /* 24 -- 39: NVMHCI */
469
    if (BITCLR(ahci->cap2, AHCI_CAP2_NVMP)) {
470 471
        g_test_message("Verifying HBA/NVMHCI area is empty.");
        for (i = AHCI_NVMHCI; i < AHCI_VENDOR; ++i) {
472
            reg = ahci_rreg(ahci, i);
473 474 475 476 477 478 479
            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) {
480
        reg = ahci_rreg(ahci, i);
481 482 483 484 485 486 487
        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);
488
            ahci_test_port_spec(ahci, i);
489 490 491 492 493 494 495 496
        } 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) {
497
                reg = ahci_rreg(ahci, j);
498 499 500 501 502 503 504 505 506
                g_assert_cmphex(reg, ==, 0);
            }
        }
    }
}

/**
 * Test the memory space for one port for specification adherence.
 */
507
static void ahci_test_port_spec(AHCIQState *ahci, uint8_t port)
508 509 510 511 512
{
    uint32_t reg;
    unsigned i;

    /* (0) CLB */
513
    reg = ahci_px_rreg(ahci, port, AHCI_PX_CLB);
514 515 516
    ASSERT_BIT_CLEAR(reg, AHCI_PX_CLB_RESERVED);

    /* (1) CLBU */
517
    if (BITCLR(ahci->cap, AHCI_CAP_S64A)) {
518
        reg = ahci_px_rreg(ahci, port, AHCI_PX_CLBU);
519 520 521 522
        g_assert_cmphex(reg, ==, 0);
    }

    /* (2) FB */
523
    reg = ahci_px_rreg(ahci, port, AHCI_PX_FB);
524 525 526
    ASSERT_BIT_CLEAR(reg, AHCI_PX_FB_RESERVED);

    /* (3) FBU */
527
    if (BITCLR(ahci->cap, AHCI_CAP_S64A)) {
528
        reg = ahci_px_rreg(ahci, port, AHCI_PX_FBU);
529 530 531 532
        g_assert_cmphex(reg, ==, 0);
    }

    /* (4) IS */
533
    reg = ahci_px_rreg(ahci, port, AHCI_PX_IS);
534 535 536
    g_assert_cmphex(reg, ==, 0);

    /* (5) IE */
537
    reg = ahci_px_rreg(ahci, port, AHCI_PX_IE);
538 539 540
    g_assert_cmphex(reg, ==, 0);

    /* (6) CMD */
541
    reg = ahci_px_rreg(ahci, port, AHCI_PX_CMD);
542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562
    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. */
563
    if (BITCLR(ahci->cap, AHCI_CAP_SMPS)) {
564 565 566 567 568 569 570 571 572 573
        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. */
    if (BITANY(reg, AHCI_PX_CMD_CPD || AHCI_PX_CMD_MPSP)) {
        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. */
574
    if (BITCLR(ahci->cap, AHCI_CAP_FBSS)) {
575 576 577 578
        ASSERT_BIT_CLEAR(reg, AHCI_PX_CMD_FBSCP);
    }

    /* (7) RESERVED */
579
    reg = ahci_px_rreg(ahci, port, AHCI_PX_RES1);
580 581 582
    g_assert_cmphex(reg, ==, 0);

    /* (8) TFD */
583
    reg = ahci_px_rreg(ahci, port, AHCI_PX_TFD);
584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600
    /* 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 */
601
    reg = ahci_px_rreg(ahci, port, AHCI_PX_SSTS);
602 603 604 605 606
    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 */
607
    reg = ahci_px_rreg(ahci, port, AHCI_PX_SCTL);
608 609 610
    g_assert_cmphex(reg, ==, 0);

    /* (12) SERR / SCR1: SError */
611
    reg = ahci_px_rreg(ahci, port, AHCI_PX_SERR);
612 613 614
    g_assert_cmphex(reg, ==, 0);

    /* (13) SACT / SCR3: SActive */
615
    reg = ahci_px_rreg(ahci, port, AHCI_PX_SACT);
616 617 618
    g_assert_cmphex(reg, ==, 0);

    /* (14) CI */
619
    reg = ahci_px_rreg(ahci, port, AHCI_PX_CI);
620 621 622
    g_assert_cmphex(reg, ==, 0);

    /* (15) SNTF */
623
    reg = ahci_px_rreg(ahci, port, AHCI_PX_SNTF);
624 625 626
    g_assert_cmphex(reg, ==, 0);

    /* (16) FBS */
627
    reg = ahci_px_rreg(ahci, port, AHCI_PX_FBS);
628 629 630 631 632 633
    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);
634
    if (BITSET(ahci->cap, AHCI_CAP_FBSS)) {
635 636 637 638 639 640
        /* 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) {
641
        reg = ahci_px_rreg(ahci, port, i);
642 643 644 645 646
        g_assert_cmphex(reg, ==, 0);
    }

    /* [28 -- 31] Vendor-Specific */
    for (i = AHCI_PX_VS; i < 32; ++i) {
647
        reg = ahci_px_rreg(ahci, port, i);
648 649 650 651 652 653
        if (reg) {
            g_test_message("INFO: Vendor register %u non-empty", i);
        }
    }
}

654 655 656 657
/**
 * Utilizing an initialized AHCI HBA, issue an IDENTIFY command to the first
 * device we see, then read and check the response.
 */
658
static void ahci_test_identify(AHCIQState *ahci)
659 660 661 662
{
    RegD2HFIS *d2h = g_malloc0(0x20);
    RegD2HFIS *pio = g_malloc0(0x20);
    RegH2DFIS fis;
663
    AHCICommandHeader cmd;
664
    PRD prd;
665
    uint32_t reg, data_ptr;
666 667 668
    uint16_t buff[256];
    unsigned i;
    int rc;
669 670
    uint8_t cx;
    uint64_t table;
671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688

    g_assert(ahci != NULL);

    /* We need to:
     * (1) Create a Command Table Buffer and update the Command List Slot #0
     *     to point to this buffer.
     * (2) Construct an FIS host-to-device command structure, and write it to
     *     the top of the command table buffer.
     * (3) Create a data buffer for the IDENTIFY response to be sent to
     * (4) Create a Physical Region Descriptor that points to the data buffer,
     *     and write it to the bottom (offset 0x80) of the command table.
     * (5) Now, PxCLB points to the command list, command 0 points to
     *     our table, and our table contains an FIS instruction and a
     *     PRD that points to our rx buffer.
     * (6) We inform the HBA via PxCI that there is a command ready in slot #0.
     */

    /* Pick the first implemented and running port */
689
    i = ahci_port_select(ahci);
690 691
    g_test_message("Selected port %u for test", i);

692 693
    /* Clear out the FIS Receive area and any pending interrupts. */
    ahci_port_clear(ahci, i);
694 695 696

    /* Create a Command Table buffer. 0x80 is the smallest with a PRDTL of 0. */
    /* We need at least one PRD, so round up to the nearest 0x80 multiple.    */
697
    table = ahci_alloc(ahci, CMD_TBL_SIZ(1));
698 699 700 701
    g_assert(table);
    ASSERT_BIT_CLEAR(table, 0x7F);

    /* Create a data buffer ... where we will dump the IDENTIFY data to. */
702
    data_ptr = ahci_alloc(ahci, 512);
703 704
    g_assert(data_ptr);

705 706 707 708 709 710 711 712
    /* pick a command slot (should be 0!) */
    cx = ahci_pick_cmd(ahci, i);

    /* Construct our Command Header (set_command_header handles endianness.) */
    memset(&cmd, 0x00, sizeof(cmd));
    cmd.flags = 5;      /* reg_h2d_fis is 5 double-words long */
    cmd.flags |= 0x400; /* clear PxTFD.STS.BSY when done */
    cmd.prdtl = 1;      /* One PRD table entry. */
713
    cmd.prdbc = 0;
714
    cmd.ctba = table;
715 716

    /* Construct our PRD, noting that DBC is 0-indexed. */
717
    prd.dba = cpu_to_le64(data_ptr);
718 719 720 721 722 723 724 725 726 727 728 729
    prd.res = 0;
    /* 511+1 bytes, request DPS interrupt */
    prd.dbc = cpu_to_le32(511 | 0x80000000);

    /* Construct our Command FIS, Based on http://wiki.osdev.org/AHCI */
    memset(&fis, 0x00, sizeof(fis));
    fis.fis_type = 0x27; /* Register Host-to-Device FIS */
    fis.command = 0xEC;  /* IDENTIFY */
    fis.device = 0;
    fis.flags = 0x80;    /* Indicate this is a command FIS */

    /* We've committed nothing yet, no interrupts should be posted yet. */
730
    g_assert_cmphex(ahci_px_rreg(ahci, i, AHCI_PX_IS), ==, 0);
731 732 733 734 735 736 737

    /* Commit the Command FIS to the Command Table */
    memwrite(table, &fis, sizeof(fis));

    /* Commit the PRD entry to the Command Table */
    memwrite(table + 0x80, &prd, sizeof(prd));

738 739
    /* Commit Command #cx, pointing to the Table, to the Command List Buffer. */
    ahci_set_command_header(ahci, i, cx, &cmd);
740

741 742
    /* Everything is in place, but we haven't given the go-ahead yet,
     * so we should find that there are no pending interrupts yet. */
743
    g_assert_cmphex(ahci_px_rreg(ahci, i, AHCI_PX_IS), ==, 0);
744

745 746
    /* Issue Command #cx via PxCI */
    ahci_px_wreg(ahci, i, AHCI_PX_CI, (1 << cx));
747
    while (BITSET(ahci_px_rreg(ahci, i, AHCI_PX_TFD), AHCI_PX_TFD_STS_BSY)) {
748 749
        usleep(50);
    }
750
    /* Check registers for post-command consistency */
751
    ahci_port_check_error(ahci, i);
752
    /* BUG: we expect AHCI_PX_IS_DPS to be set. */
753
    ahci_port_check_interrupts(ahci, i, AHCI_PX_IS_DHRS | AHCI_PX_IS_PSS);
754

755 756 757
    /* Investigate the CMD, assert that we read 512 bytes */
    ahci_get_command_header(ahci, i, cx, &cmd);
    g_assert_cmphex(512, ==, cmd.prdbc);
758 759

    /* Investigate FIS responses */
760 761
    memread(ahci->port[i].fb + 0x20, pio, 0x20);
    memread(ahci->port[i].fb + 0x40, d2h, 0x20);
762 763 764 765 766 767
    g_assert_cmphex(pio->fis_type, ==, 0x5f);
    g_assert_cmphex(d2h->fis_type, ==, 0x34);
    g_assert_cmphex(pio->flags, ==, d2h->flags);
    g_assert_cmphex(pio->status, ==, d2h->status);
    g_assert_cmphex(pio->error, ==, d2h->error);

768
    reg = ahci_px_rreg(ahci, i, AHCI_PX_TFD);
769 770 771 772 773
    g_assert_cmphex((reg & AHCI_PX_TFD_ERR), ==, pio->error);
    g_assert_cmphex((reg & AHCI_PX_TFD_STS), ==, pio->status);
    /* The PIO Setup FIS contains a "bytes read" field, which is a
     * 16-bit value. The Physical Region Descriptor Byte Count is
     * 32-bit, but for small transfers using one PRD, it should match. */
774
    g_assert_cmphex(le16_to_cpu(pio->res4), ==, cmd.prdbc);
775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795

    /* Last, but not least: Investigate the IDENTIFY response data. */
    memread(data_ptr, &buff, 512);

    /* 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);

    g_free(d2h);
    g_free(pio);
}

796 797 798 799 800 801 802 803 804
/******************************************************************************/
/* Test Interfaces                                                            */
/******************************************************************************/

/**
 * Basic sanity test to boot a machine, find an AHCI device, and shutdown.
 */
static void test_sanity(void)
{
J
John Snow 已提交
805
    AHCIQState *ahci;
806 807 808 809
    ahci = ahci_boot();
    ahci_shutdown(ahci);
}

810 811 812 813 814 815
/**
 * 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 已提交
816
    AHCIQState *ahci;
817
    ahci = ahci_boot();
818
    ahci_test_pci_spec(ahci);
819 820 821
    ahci_shutdown(ahci);
}

822 823 824 825 826 827
/**
 * 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 已提交
828
    AHCIQState *ahci;
829

830
    ahci = ahci_boot();
831
    ahci_pci_enable(ahci);
832 833 834
    ahci_shutdown(ahci);
}

835 836 837 838 839 840
/**
 * Investigate the memory mapped regions of the HBA,
 * and test them for AHCI specification adherence.
 */
static void test_hba_spec(void)
{
J
John Snow 已提交
841
    AHCIQState *ahci;
842 843

    ahci = ahci_boot();
844 845
    ahci_pci_enable(ahci);
    ahci_test_hba_spec(ahci);
846 847 848
    ahci_shutdown(ahci);
}

849 850 851 852 853 854
/**
 * 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 已提交
855
    AHCIQState *ahci;
856 857

    ahci = ahci_boot();
858 859
    ahci_pci_enable(ahci);
    ahci_hba_enable(ahci);
860 861 862
    ahci_shutdown(ahci);
}

863 864 865 866 867 868
/**
 * 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 已提交
869
    AHCIQState *ahci;
870 871

    ahci = ahci_boot();
872 873 874
    ahci_pci_enable(ahci);
    ahci_hba_enable(ahci);
    ahci_test_identify(ahci);
875 876 877
    ahci_shutdown(ahci);
}

878 879 880 881 882 883 884
/******************************************************************************/

int main(int argc, char **argv)
{
    const char *arch;
    int fd;
    int ret;
885 886 887 888 889 890
    int c;

    static struct option long_options[] = {
        {"pedantic", no_argument, 0, 'p' },
        {0, 0, 0, 0},
    };
891 892 893 894

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

895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911
    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();
        }
    }

912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927
    /* Check architecture */
    arch = qtest_get_arch();
    if (strcmp(arch, "i386") && strcmp(arch, "x86_64")) {
        g_test_message("Skipping test for non-x86");
        return 0;
    }

    /* Create a temporary raw image */
    fd = mkstemp(tmp_path);
    g_assert(fd >= 0);
    ret = ftruncate(fd, TEST_IMAGE_SIZE);
    g_assert(ret == 0);
    close(fd);

    /* Run the tests */
    qtest_add_func("/ahci/sanity",     test_sanity);
928
    qtest_add_func("/ahci/pci_spec",   test_pci_spec);
929
    qtest_add_func("/ahci/pci_enable", test_pci_enable);
930
    qtest_add_func("/ahci/hba_spec",   test_hba_spec);
931
    qtest_add_func("/ahci/hba_enable", test_hba_enable);
932
    qtest_add_func("/ahci/identify",   test_identify);
933 934 935 936 937 938 939 940

    ret = g_test_run();

    /* Cleanup */
    unlink(tmp_path);

    return ret;
}