| 
									
										
										
										
											2016-02-18 14:16:18 +00: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"
 | 
					
						
							| 
									
										
										
										
											2020-10-06 15:38:55 +03:00
										 |  |  | #include "qapi/error.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-08 13:48:08 -03:00
										 |  |  | #include "trace.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline const char *sdbus_name(SDBus *sdbus) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return sdbus->qbus.name; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-02-18 14:16:18 +00: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 13:48:10 -03: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 13:48:09 -03: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 14:16:18 +00:00
										 |  |  | int sdbus_do_command(SDBus *sdbus, SDRequest *req, uint8_t *response) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     SDState *card = get_card(sdbus); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-29 15:11:19 +01:00
										 |  |  |     trace_sdbus_command(sdbus_name(sdbus), req->cmd, req->arg); | 
					
						
							| 
									
										
										
										
											2016-02-18 14:16:18 +00: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 14:16:18 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     SDState *card = get_card(sdbus); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-08 13:48:08 -03:00
										 |  |  |     trace_sdbus_write(sdbus_name(sdbus), value); | 
					
						
							| 
									
										
										
										
											2016-02-18 14:16:18 +00: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 14:16:18 +00: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 14:16:18 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     SDState *card = get_card(sdbus); | 
					
						
							| 
									
										
										
										
											2018-02-08 13:48:08 -03:00
										 |  |  |     uint8_t value = 0; | 
					
						
							| 
									
										
										
										
											2016-02-18 14:16:18 +00: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 14:16:18 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-02-08 13:48:08 -03:00
										 |  |  |     trace_sdbus_read(sdbus_name(sdbus), value); | 
					
						
							| 
									
										
										
										
											2016-02-18 14:16:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-08 13:48:08 -03:00
										 |  |  |     return value; | 
					
						
							| 
									
										
										
										
											2016-02-18 14:16:18 +00: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]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-28 14:30:30 +08:00
										 |  |  | bool sdbus_receive_ready(SDBus *sdbus) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     SDState *card = get_card(sdbus); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (card) { | 
					
						
							|  |  |  |         SDCardClass *sc = SD_CARD_GET_CLASS(card); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return sc->receive_ready(card); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-18 14:16:18 +00: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 14:55:09 +00: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); | 
					
						
							| 
									
										
										
										
											2020-10-06 15:38:55 +03:00
										 |  |  |     qdev_set_parent_bus(DEVICE(card), &to->qbus, &error_abort); | 
					
						
							| 
									
										
										
										
											2017-02-28 14:55:09 +00:00
										 |  |  |     sdbus_set_inserted(to, true); | 
					
						
							|  |  |  |     sdbus_set_readonly(to, readonly); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-18 14:16:18 +00: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) |