balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Intel XScale PXA255/270 PC Card and CompactFlash Interface. |
| 3 | * |
| 4 | * Copyright (c) 2006 Openedhand Ltd. |
| 5 | * Written by Andrzej Zaborowski <balrog@zabor.org> |
| 6 | * |
| 7 | * This code is licensed under the GPLv2. |
Paolo Bonzini | 6b620ca | 2012-01-13 17:44:23 +0100 | [diff] [blame] | 8 | * |
| 9 | * Contributions after 2012-01-13 are licensed under the terms of the |
| 10 | * GNU GPL, version 2 or (at your option) any later version. |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 11 | */ |
| 12 | |
pbrook | 87ecb68 | 2007-11-17 17:14:51 +0000 | [diff] [blame] | 13 | #include "hw.h" |
| 14 | #include "pcmcia.h" |
pbrook | 9596ebb | 2007-11-18 01:44:38 +0000 | [diff] [blame] | 15 | #include "pxa.h" |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 16 | |
Benoît Canet | 4beeaa7 | 2011-10-30 14:50:13 +0100 | [diff] [blame] | 17 | |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 18 | struct PXA2xxPCMCIAState { |
| 19 | PCMCIASocket slot; |
| 20 | PCMCIACardState *card; |
Benoît Canet | 354a8c0 | 2011-10-30 14:50:12 +0100 | [diff] [blame] | 21 | MemoryRegion common_iomem; |
Benoît Canet | 4beeaa7 | 2011-10-30 14:50:13 +0100 | [diff] [blame] | 22 | MemoryRegion attr_iomem; |
Benoît Canet | 59aee13 | 2011-10-30 14:50:14 +0100 | [diff] [blame] | 23 | MemoryRegion iomem; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 24 | |
| 25 | qemu_irq irq; |
| 26 | qemu_irq cd_irq; |
| 27 | }; |
| 28 | |
Benoît Canet | 354a8c0 | 2011-10-30 14:50:12 +0100 | [diff] [blame] | 29 | static uint64_t pxa2xx_pcmcia_common_read(void *opaque, |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 30 | hwaddr offset, unsigned size) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 31 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 32 | PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 33 | |
| 34 | if (s->slot.attached) { |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 35 | return s->card->common_read(s->card->state, offset); |
| 36 | } |
| 37 | |
| 38 | return 0; |
| 39 | } |
| 40 | |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 41 | static void pxa2xx_pcmcia_common_write(void *opaque, hwaddr offset, |
Benoît Canet | 354a8c0 | 2011-10-30 14:50:12 +0100 | [diff] [blame] | 42 | uint64_t value, unsigned size) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 43 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 44 | PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 45 | |
| 46 | if (s->slot.attached) { |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 47 | s->card->common_write(s->card->state, offset, value); |
| 48 | } |
| 49 | } |
| 50 | |
Benoît Canet | 4beeaa7 | 2011-10-30 14:50:13 +0100 | [diff] [blame] | 51 | static uint64_t pxa2xx_pcmcia_attr_read(void *opaque, |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 52 | hwaddr offset, unsigned size) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 53 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 54 | PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 55 | |
| 56 | if (s->slot.attached) { |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 57 | return s->card->attr_read(s->card->state, offset); |
| 58 | } |
| 59 | |
| 60 | return 0; |
| 61 | } |
| 62 | |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 63 | static void pxa2xx_pcmcia_attr_write(void *opaque, hwaddr offset, |
Benoît Canet | 4beeaa7 | 2011-10-30 14:50:13 +0100 | [diff] [blame] | 64 | uint64_t value, unsigned size) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 65 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 66 | PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 67 | |
| 68 | if (s->slot.attached) { |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 69 | s->card->attr_write(s->card->state, offset, value); |
| 70 | } |
| 71 | } |
| 72 | |
Benoît Canet | 59aee13 | 2011-10-30 14:50:14 +0100 | [diff] [blame] | 73 | static uint64_t pxa2xx_pcmcia_io_read(void *opaque, |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 74 | hwaddr offset, unsigned size) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 75 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 76 | PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 77 | |
| 78 | if (s->slot.attached) { |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 79 | return s->card->io_read(s->card->state, offset); |
| 80 | } |
| 81 | |
| 82 | return 0; |
| 83 | } |
| 84 | |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 85 | static void pxa2xx_pcmcia_io_write(void *opaque, hwaddr offset, |
Benoît Canet | 59aee13 | 2011-10-30 14:50:14 +0100 | [diff] [blame] | 86 | uint64_t value, unsigned size) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 87 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 88 | PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 89 | |
| 90 | if (s->slot.attached) { |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 91 | s->card->io_write(s->card->state, offset, value); |
| 92 | } |
| 93 | } |
| 94 | |
Benoît Canet | 354a8c0 | 2011-10-30 14:50:12 +0100 | [diff] [blame] | 95 | static const MemoryRegionOps pxa2xx_pcmcia_common_ops = { |
| 96 | .read = pxa2xx_pcmcia_common_read, |
| 97 | .write = pxa2xx_pcmcia_common_write, |
| 98 | .endianness = DEVICE_NATIVE_ENDIAN |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 99 | }; |
| 100 | |
Benoît Canet | 4beeaa7 | 2011-10-30 14:50:13 +0100 | [diff] [blame] | 101 | static const MemoryRegionOps pxa2xx_pcmcia_attr_ops = { |
| 102 | .read = pxa2xx_pcmcia_attr_read, |
| 103 | .write = pxa2xx_pcmcia_attr_write, |
| 104 | .endianness = DEVICE_NATIVE_ENDIAN |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 105 | }; |
| 106 | |
Benoît Canet | 59aee13 | 2011-10-30 14:50:14 +0100 | [diff] [blame] | 107 | static const MemoryRegionOps pxa2xx_pcmcia_io_ops = { |
| 108 | .read = pxa2xx_pcmcia_io_read, |
| 109 | .write = pxa2xx_pcmcia_io_write, |
| 110 | .endianness = DEVICE_NATIVE_ENDIAN |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 111 | }; |
| 112 | |
| 113 | static void pxa2xx_pcmcia_set_irq(void *opaque, int line, int level) |
| 114 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 115 | PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 116 | if (!s->irq) |
| 117 | return; |
| 118 | |
| 119 | qemu_set_irq(s->irq, level); |
| 120 | } |
| 121 | |
Benoît Canet | 354a8c0 | 2011-10-30 14:50:12 +0100 | [diff] [blame] | 122 | PXA2xxPCMCIAState *pxa2xx_pcmcia_init(MemoryRegion *sysmem, |
Avi Kivity | a8170e5 | 2012-10-23 12:30:10 +0200 | [diff] [blame] | 123 | hwaddr base) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 124 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 125 | PXA2xxPCMCIAState *s; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 126 | |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 127 | s = (PXA2xxPCMCIAState *) |
Anthony Liguori | 7267c09 | 2011-08-20 22:09:37 -0500 | [diff] [blame] | 128 | g_malloc0(sizeof(PXA2xxPCMCIAState)); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 129 | |
| 130 | /* Socket I/O Memory Space */ |
Benoît Canet | 59aee13 | 2011-10-30 14:50:14 +0100 | [diff] [blame] | 131 | memory_region_init_io(&s->iomem, &pxa2xx_pcmcia_io_ops, s, |
| 132 | "pxa2xx-pcmcia-io", 0x04000000); |
| 133 | memory_region_add_subregion(sysmem, base | 0x00000000, |
| 134 | &s->iomem); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 135 | |
| 136 | /* Then next 64 MB is reserved */ |
| 137 | |
| 138 | /* Socket Attribute Memory Space */ |
Benoît Canet | 4beeaa7 | 2011-10-30 14:50:13 +0100 | [diff] [blame] | 139 | memory_region_init_io(&s->attr_iomem, &pxa2xx_pcmcia_attr_ops, s, |
| 140 | "pxa2xx-pcmcia-attribute", 0x04000000); |
| 141 | memory_region_add_subregion(sysmem, base | 0x08000000, |
| 142 | &s->attr_iomem); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 143 | |
| 144 | /* Socket Common Memory Space */ |
Benoît Canet | 354a8c0 | 2011-10-30 14:50:12 +0100 | [diff] [blame] | 145 | memory_region_init_io(&s->common_iomem, &pxa2xx_pcmcia_common_ops, s, |
| 146 | "pxa2xx-pcmcia-common", 0x04000000); |
| 147 | memory_region_add_subregion(sysmem, base | 0x0c000000, |
| 148 | &s->common_iomem); |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 149 | |
| 150 | if (base == 0x30000000) |
| 151 | s->slot.slot_string = "PXA PC Card Socket 1"; |
| 152 | else |
| 153 | s->slot.slot_string = "PXA PC Card Socket 0"; |
| 154 | s->slot.irq = qemu_allocate_irqs(pxa2xx_pcmcia_set_irq, s, 1)[0]; |
| 155 | pcmcia_socket_register(&s->slot); |
balrog | 3f58226 | 2007-05-23 21:47:51 +0000 | [diff] [blame] | 156 | |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 157 | return s; |
| 158 | } |
| 159 | |
| 160 | /* Insert a new card into a slot */ |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 161 | int pxa2xx_pcmcia_attach(void *opaque, PCMCIACardState *card) |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 162 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 163 | PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 164 | if (s->slot.attached) |
| 165 | return -EEXIST; |
| 166 | |
| 167 | if (s->cd_irq) { |
| 168 | qemu_irq_raise(s->cd_irq); |
| 169 | } |
| 170 | |
| 171 | s->card = card; |
| 172 | |
| 173 | s->slot.attached = 1; |
| 174 | s->card->slot = &s->slot; |
| 175 | s->card->attach(s->card->state); |
| 176 | |
| 177 | return 0; |
| 178 | } |
| 179 | |
| 180 | /* Eject card from the slot */ |
| 181 | int pxa2xx_pcmcia_dettach(void *opaque) |
| 182 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 183 | PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 184 | if (!s->slot.attached) |
| 185 | return -ENOENT; |
| 186 | |
| 187 | s->card->detach(s->card->state); |
Blue Swirl | b9d38e9 | 2009-09-21 18:11:34 +0000 | [diff] [blame] | 188 | s->card->slot = NULL; |
| 189 | s->card = NULL; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 190 | |
| 191 | s->slot.attached = 0; |
| 192 | |
| 193 | if (s->irq) |
| 194 | qemu_irq_lower(s->irq); |
| 195 | if (s->cd_irq) |
| 196 | qemu_irq_lower(s->cd_irq); |
| 197 | |
| 198 | return 0; |
| 199 | } |
| 200 | |
| 201 | /* Who to notify on card events */ |
| 202 | void pxa2xx_pcmcia_set_irq_cb(void *opaque, qemu_irq irq, qemu_irq cd_irq) |
| 203 | { |
Paul Brook | bc24a22 | 2009-05-10 01:44:56 +0100 | [diff] [blame] | 204 | PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque; |
balrog | a171fe3 | 2007-04-30 01:48:07 +0000 | [diff] [blame] | 205 | s->irq = irq; |
| 206 | s->cd_irq = cd_irq; |
| 207 | } |