2016-02-18 15:16:18 +01:00
|
|
|
/*
|
|
|
|
* SD card bus interface code.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2015 Linaro Limited
|
|
|
|
*
|
|
|
|
* Author:
|
|
|
|
* Peter Maydell <peter.maydell@linaro.org>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify it
|
|
|
|
* under the terms and conditions of the GNU General Public License,
|
|
|
|
* version 2 or later, as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
|
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
|
|
|
* more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along with
|
|
|
|
* this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "qemu/osdep.h"
|
|
|
|
#include "hw/qdev-core.h"
|
|
|
|
#include "hw/sd/sd.h"
|
2019-05-23 16:35:07 +02:00
|
|
|
#include "qemu/module.h"
|
2018-02-08 17:48:08 +01:00
|
|
|
#include "trace.h"
|
|
|
|
|
|
|
|
static inline const char *sdbus_name(SDBus *sdbus)
|
|
|
|
{
|
|
|
|
return sdbus->qbus.name;
|
|
|
|
}
|
2016-02-18 15:16:18 +01:00
|
|
|
|
|
|
|
static SDState *get_card(SDBus *sdbus)
|
|
|
|
{
|
|
|
|
/* We only ever have one child on the bus so just return it */
|
|
|
|
BusChild *kid = QTAILQ_FIRST(&sdbus->qbus.children);
|
|
|
|
|
|
|
|
if (!kid) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return SD_CARD(kid->child);
|
|
|
|
}
|
|
|
|
|
2018-02-08 17:48:10 +01:00
|
|
|
uint8_t sdbus_get_dat_lines(SDBus *sdbus)
|
|
|
|
{
|
|
|
|
SDState *slave = get_card(sdbus);
|
|
|
|
uint8_t dat_lines = 0b1111; /* 4 bit bus width */
|
|
|
|
|
|
|
|
if (slave) {
|
|
|
|
SDCardClass *sc = SD_CARD_GET_CLASS(slave);
|
|
|
|
|
|
|
|
if (sc->get_dat_lines) {
|
|
|
|
dat_lines = sc->get_dat_lines(slave);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
trace_sdbus_get_dat_lines(sdbus_name(sdbus), dat_lines);
|
|
|
|
|
|
|
|
return dat_lines;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool sdbus_get_cmd_line(SDBus *sdbus)
|
|
|
|
{
|
|
|
|
SDState *slave = get_card(sdbus);
|
|
|
|
bool cmd_line = true;
|
|
|
|
|
|
|
|
if (slave) {
|
|
|
|
SDCardClass *sc = SD_CARD_GET_CLASS(slave);
|
|
|
|
|
|
|
|
if (sc->get_cmd_line) {
|
|
|
|
cmd_line = sc->get_cmd_line(slave);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
trace_sdbus_get_cmd_line(sdbus_name(sdbus), cmd_line);
|
|
|
|
|
|
|
|
return cmd_line;
|
|
|
|
}
|
|
|
|
|
2018-02-08 17:48:09 +01:00
|
|
|
void sdbus_set_voltage(SDBus *sdbus, uint16_t millivolts)
|
|
|
|
{
|
|
|
|
SDState *card = get_card(sdbus);
|
|
|
|
|
|
|
|
trace_sdbus_set_voltage(sdbus_name(sdbus), millivolts);
|
|
|
|
if (card) {
|
|
|
|
SDCardClass *sc = SD_CARD_GET_CLASS(card);
|
|
|
|
|
|
|
|
assert(sc->set_voltage);
|
|
|
|
sc->set_voltage(card, millivolts);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-18 15:16:18 +01:00
|
|
|
int sdbus_do_command(SDBus *sdbus, SDRequest *req, uint8_t *response)
|
|
|
|
{
|
|
|
|
SDState *card = get_card(sdbus);
|
|
|
|
|
2018-06-29 16:11:19 +02:00
|
|
|
trace_sdbus_command(sdbus_name(sdbus), req->cmd, req->arg);
|
2016-02-18 15:16:18 +01:00
|
|
|
if (card) {
|
|
|
|
SDCardClass *sc = SD_CARD_GET_CLASS(card);
|
|
|
|
|
|
|
|
return sc->do_command(card, req, response);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-08-14 11:23:41 +02:00
|
|
|
void sdbus_write_byte(SDBus *sdbus, uint8_t value)
|
2016-02-18 15:16:18 +01:00
|
|
|
{
|
|
|
|
SDState *card = get_card(sdbus);
|
|
|
|
|
2018-02-08 17:48:08 +01:00
|
|
|
trace_sdbus_write(sdbus_name(sdbus), value);
|
2016-02-18 15:16:18 +01:00
|
|
|
if (card) {
|
|
|
|
SDCardClass *sc = SD_CARD_GET_CLASS(card);
|
|
|
|
|
2020-08-14 11:23:40 +02:00
|
|
|
sc->write_byte(card, value);
|
2016-02-18 15:16:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-14 11:23:43 +02:00
|
|
|
void sdbus_write_data(SDBus *sdbus, const void *buf, size_t length)
|
|
|
|
{
|
|
|
|
SDState *card = get_card(sdbus);
|
|
|
|
const uint8_t *data = buf;
|
|
|
|
|
|
|
|
if (card) {
|
|
|
|
SDCardClass *sc = SD_CARD_GET_CLASS(card);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < length; i++) {
|
|
|
|
trace_sdbus_write(sdbus_name(sdbus), data[i]);
|
|
|
|
sc->write_byte(card, data[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-14 11:23:42 +02:00
|
|
|
uint8_t sdbus_read_byte(SDBus *sdbus)
|
2016-02-18 15:16:18 +01:00
|
|
|
{
|
|
|
|
SDState *card = get_card(sdbus);
|
2018-02-08 17:48:08 +01:00
|
|
|
uint8_t value = 0;
|
2016-02-18 15:16:18 +01:00
|
|
|
|
|
|
|
if (card) {
|
|
|
|
SDCardClass *sc = SD_CARD_GET_CLASS(card);
|
|
|
|
|
2020-08-14 11:23:40 +02:00
|
|
|
value = sc->read_byte(card);
|
2016-02-18 15:16:18 +01:00
|
|
|
}
|
2018-02-08 17:48:08 +01:00
|
|
|
trace_sdbus_read(sdbus_name(sdbus), value);
|
2016-02-18 15:16:18 +01:00
|
|
|
|
2018-02-08 17:48:08 +01:00
|
|
|
return value;
|
2016-02-18 15:16:18 +01:00
|
|
|
}
|
|
|
|
|
2020-08-14 11:23:45 +02:00
|
|
|
void sdbus_read_data(SDBus *sdbus, void *buf, size_t length)
|
|
|
|
{
|
|
|
|
SDState *card = get_card(sdbus);
|
|
|
|
uint8_t *data = buf;
|
|
|
|
|
|
|
|
if (card) {
|
|
|
|
SDCardClass *sc = SD_CARD_GET_CLASS(card);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < length; i++) {
|
|
|
|
data[i] = sc->read_byte(card);
|
|
|
|
trace_sdbus_read(sdbus_name(sdbus), data[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-18 15:16:18 +01:00
|
|
|
bool sdbus_data_ready(SDBus *sdbus)
|
|
|
|
{
|
|
|
|
SDState *card = get_card(sdbus);
|
|
|
|
|
|
|
|
if (card) {
|
|
|
|
SDCardClass *sc = SD_CARD_GET_CLASS(card);
|
|
|
|
|
|
|
|
return sc->data_ready(card);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool sdbus_get_inserted(SDBus *sdbus)
|
|
|
|
{
|
|
|
|
SDState *card = get_card(sdbus);
|
|
|
|
|
|
|
|
if (card) {
|
|
|
|
SDCardClass *sc = SD_CARD_GET_CLASS(card);
|
|
|
|
|
|
|
|
return sc->get_inserted(card);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool sdbus_get_readonly(SDBus *sdbus)
|
|
|
|
{
|
|
|
|
SDState *card = get_card(sdbus);
|
|
|
|
|
|
|
|
if (card) {
|
|
|
|
SDCardClass *sc = SD_CARD_GET_CLASS(card);
|
|
|
|
|
|
|
|
return sc->get_readonly(card);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void sdbus_set_inserted(SDBus *sdbus, bool inserted)
|
|
|
|
{
|
|
|
|
SDBusClass *sbc = SD_BUS_GET_CLASS(sdbus);
|
|
|
|
BusState *qbus = BUS(sdbus);
|
|
|
|
|
|
|
|
if (sbc->set_inserted) {
|
|
|
|
sbc->set_inserted(qbus->parent, inserted);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void sdbus_set_readonly(SDBus *sdbus, bool readonly)
|
|
|
|
{
|
|
|
|
SDBusClass *sbc = SD_BUS_GET_CLASS(sdbus);
|
|
|
|
BusState *qbus = BUS(sdbus);
|
|
|
|
|
|
|
|
if (sbc->set_readonly) {
|
|
|
|
sbc->set_readonly(qbus->parent, readonly);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-28 15:55:09 +01:00
|
|
|
void sdbus_reparent_card(SDBus *from, SDBus *to)
|
|
|
|
{
|
|
|
|
SDState *card = get_card(from);
|
|
|
|
SDCardClass *sc;
|
|
|
|
bool readonly;
|
|
|
|
|
|
|
|
/* We directly reparent the card object rather than implementing this
|
|
|
|
* as a hotpluggable connection because we don't want to expose SD cards
|
|
|
|
* to users as being hotpluggable, and we can get away with it in this
|
|
|
|
* limited use case. This could perhaps be implemented more cleanly in
|
|
|
|
* future by adding support to the hotplug infrastructure for "device
|
|
|
|
* can be hotplugged only via code, not by user".
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!card) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
sc = SD_CARD_GET_CLASS(card);
|
|
|
|
readonly = sc->get_readonly(card);
|
|
|
|
|
|
|
|
sdbus_set_inserted(from, false);
|
|
|
|
qdev_set_parent_bus(DEVICE(card), &to->qbus);
|
|
|
|
sdbus_set_inserted(to, true);
|
|
|
|
sdbus_set_readonly(to, readonly);
|
|
|
|
}
|
|
|
|
|
2016-02-18 15:16:18 +01:00
|
|
|
static const TypeInfo sd_bus_info = {
|
|
|
|
.name = TYPE_SD_BUS,
|
|
|
|
.parent = TYPE_BUS,
|
|
|
|
.instance_size = sizeof(SDBus),
|
|
|
|
.class_size = sizeof(SDBusClass),
|
|
|
|
};
|
|
|
|
|
|
|
|
static void sd_bus_register_types(void)
|
|
|
|
{
|
|
|
|
type_register_static(&sd_bus_info);
|
|
|
|
}
|
|
|
|
|
|
|
|
type_init(sd_bus_register_types)
|