aboutsummaryrefslogtreecommitdiff
path: root/tests/qtest/libqos/sdhci-cmd.c
blob: 2d9e51834114142a2c0a881dafd313660ff94c1c (plain)
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
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
/*
 * MMC Host Controller Commands
 *
 * Copyright (c) 2021 Google LLC
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License as published by the
 * Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
 * for more details.
 */

#include "qemu/osdep.h"
#include "sdhci-cmd.h"
#include "libqtest.h"

static ssize_t read_fifo(QTestState *qts, uint64_t reg, char *msg, size_t count)
{
    uint32_t mask = 0xff;
    size_t index = 0;
    uint32_t msg_frag;
    int size;
    while (index < count) {
        size = count - index;
        if (size > 4) {
            size = 4;
        }
        msg_frag = qtest_readl(qts, reg);
        while (size > 0) {
            msg[index] = msg_frag & mask;
            if (msg[index++] == 0) {
                return index;
            }
            msg_frag >>= 8;
            --size;
        }
    }
    return index;
}

static void write_fifo(QTestState *qts, uint64_t reg, const char *msg,
                       size_t count)
{
    size_t index = 0;
    uint32_t msg_frag;
    int size;
    int frag_i;
    while (index < count) {
        size = count - index;
        if (size > 4) {
            size = 4;
        }
        msg_frag = 0;
        frag_i = 0;
        while (frag_i < size) {
            msg_frag |= ((uint32_t)msg[index++]) << (frag_i * 8);
            ++frag_i;
        }
        qtest_writel(qts, reg, msg_frag);
    }
}

static void fill_block(QTestState *qts, uint64_t reg, int count)
{
    while (--count >= 0) {
        qtest_writel(qts, reg, 0);
    }
}

void sdhci_cmd_regs(QTestState *qts, uint64_t base_addr, uint16_t blksize,
                    uint16_t blkcnt, uint32_t argument, uint16_t trnmod,
                    uint16_t cmdreg)
{
    qtest_writew(qts, base_addr + SDHC_BLKSIZE, blksize);
    qtest_writew(qts, base_addr + SDHC_BLKCNT, blkcnt);
    qtest_writel(qts, base_addr + SDHC_ARGUMENT, argument);
    qtest_writew(qts, base_addr + SDHC_TRNMOD, trnmod);
    qtest_writew(qts, base_addr + SDHC_CMDREG, cmdreg);
}

ssize_t sdhci_read_cmd(QTestState *qts, uint64_t base_addr, char *msg,
                       size_t count)
{
    sdhci_cmd_regs(qts, base_addr, count, 1, 0,
                   SDHC_TRNS_MULTI | SDHC_TRNS_READ | SDHC_TRNS_BLK_CNT_EN,
                   SDHC_READ_MULTIPLE_BLOCK | SDHC_CMD_DATA_PRESENT);

    /* read sd fifo_buffer */
    ssize_t bytes_read = read_fifo(qts, base_addr + SDHC_BDATA, msg, count);

    sdhci_cmd_regs(qts, base_addr, 0, 0, 0,
                   SDHC_TRNS_MULTI | SDHC_TRNS_READ | SDHC_TRNS_BLK_CNT_EN,
                   SDHC_STOP_TRANSMISSION);

    return bytes_read;
}

void sdhci_write_cmd(QTestState *qts, uint64_t base_addr, const char *msg,
                     size_t count, size_t blksize)
{
    sdhci_cmd_regs(qts, base_addr, blksize, 1, 0,
                   SDHC_TRNS_MULTI | SDHC_TRNS_WRITE | SDHC_TRNS_BLK_CNT_EN,
                   SDHC_WRITE_MULTIPLE_BLOCK | SDHC_CMD_DATA_PRESENT);

    /* write to sd fifo_buffer */
    write_fifo(qts, base_addr + SDHC_BDATA, msg, count);
    fill_block(qts, base_addr + SDHC_BDATA, (blksize - count) / 4);

    sdhci_cmd_regs(qts, base_addr, 0, 0, 0,
                   SDHC_TRNS_MULTI | SDHC_TRNS_WRITE | SDHC_TRNS_BLK_CNT_EN,
                   SDHC_STOP_TRANSMISSION);
}