1 /* 2 * SD card bus interface code. 3 * 4 * Copyright (c) 2015 Linaro Limited 5 * 6 * Author: 7 * Peter Maydell <peter.maydell@linaro.org> 8 * 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. 12 * 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. 17 * 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/>. 20 */ 21 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" 27 28 static inline const char *sdbus_name(SDBus *sdbus) 29 { 30 return sdbus->qbus.name; 31 } 32 33 static SDState *get_card(SDBus *sdbus) 34 { 35 /* We only ever have one child on the bus so just return it */ 36 BusChild *kid = QTAILQ_FIRST(&sdbus->qbus.children); 37 38 if (!kid) { 39 return NULL; 40 } 41 return SD_CARD(kid->child); 42 } 43 44 uint8_t sdbus_get_dat_lines(SDBus *sdbus) 45 { 46 SDState *slave = get_card(sdbus); 47 uint8_t dat_lines = 0b1111; /* 4 bit bus width */ 48 49 if (slave) { 50 SDCardClass *sc = SD_CARD_GET_CLASS(slave); 51 52 if (sc->get_dat_lines) { 53 dat_lines = sc->get_dat_lines(slave); 54 } 55 } 56 trace_sdbus_get_dat_lines(sdbus_name(sdbus), dat_lines); 57 58 return dat_lines; 59 } 60 61 bool sdbus_get_cmd_line(SDBus *sdbus) 62 { 63 SDState *slave = get_card(sdbus); 64 bool cmd_line = true; 65 66 if (slave) { 67 SDCardClass *sc = SD_CARD_GET_CLASS(slave); 68 69 if (sc->get_cmd_line) { 70 cmd_line = sc->get_cmd_line(slave); 71 } 72 } 73 trace_sdbus_get_cmd_line(sdbus_name(sdbus), cmd_line); 74 75 return cmd_line; 76 } 77 78 void sdbus_set_voltage(SDBus *sdbus, uint16_t millivolts) 79 { 80 SDState *card = get_card(sdbus); 81 82 trace_sdbus_set_voltage(sdbus_name(sdbus), millivolts); 83 if (card) { 84 SDCardClass *sc = SD_CARD_GET_CLASS(card); 85 86 assert(sc->set_voltage); 87 sc->set_voltage(card, millivolts); 88 } 89 } 90 91 int sdbus_do_command(SDBus *sdbus, SDRequest *req, uint8_t *response) 92 { 93 SDState *card = get_card(sdbus); 94 95 trace_sdbus_command(sdbus_name(sdbus), req->cmd, req->arg); 96 if (card) { 97 SDCardClass *sc = SD_CARD_GET_CLASS(card); 98 99 return sc->do_command(card, req, response); 100 } 101 102 return 0; 103 } 104 105 void sdbus_write_byte(SDBus *sdbus, uint8_t value) 106 { 107 SDState *card = get_card(sdbus); 108 109 trace_sdbus_write(sdbus_name(sdbus), value); 110 if (card) { 111 SDCardClass *sc = SD_CARD_GET_CLASS(card); 112 113 sc->write_byte(card, value); 114 } 115 } 116 117 void sdbus_write_data(SDBus *sdbus, const void *buf, size_t length) 118 { 119 SDState *card = get_card(sdbus); 120 const uint8_t *data = buf; 121 122 if (card) { 123 SDCardClass *sc = SD_CARD_GET_CLASS(card); 124 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]); 128 } 129 } 130 } 131 132 uint8_t sdbus_read_byte(SDBus *sdbus) 133 { 134 SDState *card = get_card(sdbus); 135 uint8_t value = 0; 136 137 if (card) { 138 SDCardClass *sc = SD_CARD_GET_CLASS(card); 139 140 value = sc->read_byte(card); 141 } 142 trace_sdbus_read(sdbus_name(sdbus), value); 143 144 return value; 145 } 146 147 void sdbus_read_data(SDBus *sdbus, void *buf, size_t length) 148 { 149 SDState *card = get_card(sdbus); 150 uint8_t *data = buf; 151 152 if (card) { 153 SDCardClass *sc = SD_CARD_GET_CLASS(card); 154 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]); 158 } 159 } 160 } 161 162 bool sdbus_data_ready(SDBus *sdbus) 163 { 164 SDState *card = get_card(sdbus); 165 166 if (card) { 167 SDCardClass *sc = SD_CARD_GET_CLASS(card); 168 169 return sc->data_ready(card); 170 } 171 172 return false; 173 } 174 175 bool sdbus_get_inserted(SDBus *sdbus) 176 { 177 SDState *card = get_card(sdbus); 178 179 if (card) { 180 SDCardClass *sc = SD_CARD_GET_CLASS(card); 181 182 return sc->get_inserted(card); 183 } 184 185 return false; 186 } 187 188 bool sdbus_get_readonly(SDBus *sdbus) 189 { 190 SDState *card = get_card(sdbus); 191 192 if (card) { 193 SDCardClass *sc = SD_CARD_GET_CLASS(card); 194 195 return sc->get_readonly(card); 196 } 197 198 return false; 199 } 200 201 void sdbus_set_inserted(SDBus *sdbus, bool inserted) 202 { 203 SDBusClass *sbc = SD_BUS_GET_CLASS(sdbus); 204 BusState *qbus = BUS(sdbus); 205 206 if (sbc->set_inserted) { 207 sbc->set_inserted(qbus->parent, inserted); 208 } 209 } 210 211 void sdbus_set_readonly(SDBus *sdbus, bool readonly) 212 { 213 SDBusClass *sbc = SD_BUS_GET_CLASS(sdbus); 214 BusState *qbus = BUS(sdbus); 215 216 if (sbc->set_readonly) { 217 sbc->set_readonly(qbus->parent, readonly); 218 } 219 } 220 221 void sdbus_reparent_card(SDBus *from, SDBus *to) 222 { 223 SDState *card = get_card(from); 224 SDCardClass *sc; 225 bool readonly; 226 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". 233 */ 234 235 if (!card) { 236 return; 237 } 238 239 sc = SD_CARD_GET_CLASS(card); 240 readonly = sc->get_readonly(card); 241 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); 246 } 247 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), 253 }; 254 255 static void sd_bus_register_types(void) 256 { 257 type_register_static(&sd_bus_info); 258 } 259 260 type_init(sd_bus_register_types) 261