virtio-ccw-input: fix description
[qemu/ar7.git] / hw / sd / core.c
blob957d116f1a7181448c7ee69a48e19b99d6e32657
1 /*
2 * SD card bus interface code.
4 * Copyright (c) 2015 Linaro Limited
6 * Author:
7 * Peter Maydell <peter.maydell@linaro.org>
9 * This program is free software; you can redistribute it and/or modify it
10 * under the terms and conditions of the GNU General Public License,
11 * version 2 or later, as published by the Free Software Foundation.
13 * This program is distributed in the hope it will be useful, but WITHOUT
14 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
15 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
16 * more details.
18 * You should have received a copy of the GNU General Public License along with
19 * this program. If not, see <http://www.gnu.org/licenses/>.
22 #include "qemu/osdep.h"
23 #include "hw/qdev-core.h"
24 #include "hw/sd/sd.h"
25 #include "qemu/module.h"
26 #include "trace.h"
28 static inline const char *sdbus_name(SDBus *sdbus)
30 return sdbus->qbus.name;
33 static SDState *get_card(SDBus *sdbus)
35 /* We only ever have one child on the bus so just return it */
36 BusChild *kid = QTAILQ_FIRST(&sdbus->qbus.children);
38 if (!kid) {
39 return NULL;
41 return SD_CARD(kid->child);
44 uint8_t sdbus_get_dat_lines(SDBus *sdbus)
46 SDState *slave = get_card(sdbus);
47 uint8_t dat_lines = 0b1111; /* 4 bit bus width */
49 if (slave) {
50 SDCardClass *sc = SD_CARD_GET_CLASS(slave);
52 if (sc->get_dat_lines) {
53 dat_lines = sc->get_dat_lines(slave);
56 trace_sdbus_get_dat_lines(sdbus_name(sdbus), dat_lines);
58 return dat_lines;
61 bool sdbus_get_cmd_line(SDBus *sdbus)
63 SDState *slave = get_card(sdbus);
64 bool cmd_line = true;
66 if (slave) {
67 SDCardClass *sc = SD_CARD_GET_CLASS(slave);
69 if (sc->get_cmd_line) {
70 cmd_line = sc->get_cmd_line(slave);
73 trace_sdbus_get_cmd_line(sdbus_name(sdbus), cmd_line);
75 return cmd_line;
78 void sdbus_set_voltage(SDBus *sdbus, uint16_t millivolts)
80 SDState *card = get_card(sdbus);
82 trace_sdbus_set_voltage(sdbus_name(sdbus), millivolts);
83 if (card) {
84 SDCardClass *sc = SD_CARD_GET_CLASS(card);
86 assert(sc->set_voltage);
87 sc->set_voltage(card, millivolts);
91 int sdbus_do_command(SDBus *sdbus, SDRequest *req, uint8_t *response)
93 SDState *card = get_card(sdbus);
95 trace_sdbus_command(sdbus_name(sdbus), req->cmd, req->arg);
96 if (card) {
97 SDCardClass *sc = SD_CARD_GET_CLASS(card);
99 return sc->do_command(card, req, response);
102 return 0;
105 void sdbus_write_byte(SDBus *sdbus, uint8_t value)
107 SDState *card = get_card(sdbus);
109 trace_sdbus_write(sdbus_name(sdbus), value);
110 if (card) {
111 SDCardClass *sc = SD_CARD_GET_CLASS(card);
113 sc->write_byte(card, value);
117 void sdbus_write_data(SDBus *sdbus, const void *buf, size_t length)
119 SDState *card = get_card(sdbus);
120 const uint8_t *data = buf;
122 if (card) {
123 SDCardClass *sc = SD_CARD_GET_CLASS(card);
125 for (size_t i = 0; i < length; i++) {
126 trace_sdbus_write(sdbus_name(sdbus), data[i]);
127 sc->write_byte(card, data[i]);
132 uint8_t sdbus_read_byte(SDBus *sdbus)
134 SDState *card = get_card(sdbus);
135 uint8_t value = 0;
137 if (card) {
138 SDCardClass *sc = SD_CARD_GET_CLASS(card);
140 value = sc->read_byte(card);
142 trace_sdbus_read(sdbus_name(sdbus), value);
144 return value;
147 void sdbus_read_data(SDBus *sdbus, void *buf, size_t length)
149 SDState *card = get_card(sdbus);
150 uint8_t *data = buf;
152 if (card) {
153 SDCardClass *sc = SD_CARD_GET_CLASS(card);
155 for (size_t i = 0; i < length; i++) {
156 data[i] = sc->read_byte(card);
157 trace_sdbus_read(sdbus_name(sdbus), data[i]);
162 bool sdbus_data_ready(SDBus *sdbus)
164 SDState *card = get_card(sdbus);
166 if (card) {
167 SDCardClass *sc = SD_CARD_GET_CLASS(card);
169 return sc->data_ready(card);
172 return false;
175 bool sdbus_get_inserted(SDBus *sdbus)
177 SDState *card = get_card(sdbus);
179 if (card) {
180 SDCardClass *sc = SD_CARD_GET_CLASS(card);
182 return sc->get_inserted(card);
185 return false;
188 bool sdbus_get_readonly(SDBus *sdbus)
190 SDState *card = get_card(sdbus);
192 if (card) {
193 SDCardClass *sc = SD_CARD_GET_CLASS(card);
195 return sc->get_readonly(card);
198 return false;
201 void sdbus_set_inserted(SDBus *sdbus, bool inserted)
203 SDBusClass *sbc = SD_BUS_GET_CLASS(sdbus);
204 BusState *qbus = BUS(sdbus);
206 if (sbc->set_inserted) {
207 sbc->set_inserted(qbus->parent, inserted);
211 void sdbus_set_readonly(SDBus *sdbus, bool readonly)
213 SDBusClass *sbc = SD_BUS_GET_CLASS(sdbus);
214 BusState *qbus = BUS(sdbus);
216 if (sbc->set_readonly) {
217 sbc->set_readonly(qbus->parent, readonly);
221 void sdbus_reparent_card(SDBus *from, SDBus *to)
223 SDState *card = get_card(from);
224 SDCardClass *sc;
225 bool readonly;
227 /* We directly reparent the card object rather than implementing this
228 * as a hotpluggable connection because we don't want to expose SD cards
229 * to users as being hotpluggable, and we can get away with it in this
230 * limited use case. This could perhaps be implemented more cleanly in
231 * future by adding support to the hotplug infrastructure for "device
232 * can be hotplugged only via code, not by user".
235 if (!card) {
236 return;
239 sc = SD_CARD_GET_CLASS(card);
240 readonly = sc->get_readonly(card);
242 sdbus_set_inserted(from, false);
243 qdev_set_parent_bus(DEVICE(card), &to->qbus);
244 sdbus_set_inserted(to, true);
245 sdbus_set_readonly(to, readonly);
248 static const TypeInfo sd_bus_info = {
249 .name = TYPE_SD_BUS,
250 .parent = TYPE_BUS,
251 .instance_size = sizeof(SDBus),
252 .class_size = sizeof(SDBusClass),
255 static void sd_bus_register_types(void)
257 type_register_static(&sd_bus_info);
260 type_init(sd_bus_register_types)