blob: 3a79c728ab2446e76374b9c4da63eb324032b175 [file] [log] [blame]
balroga171fe32007-04-30 01:48:07 +00001/*
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 Bonzini6b620ca2012-01-13 17:44:23 +01008 *
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.
balroga171fe32007-04-30 01:48:07 +000011 */
12
pbrook87ecb682007-11-17 17:14:51 +000013#include "hw.h"
14#include "pcmcia.h"
pbrook9596ebb2007-11-18 01:44:38 +000015#include "pxa.h"
balroga171fe32007-04-30 01:48:07 +000016
Benoît Canet4beeaa72011-10-30 14:50:13 +010017
Paul Brookbc24a222009-05-10 01:44:56 +010018struct PXA2xxPCMCIAState {
19 PCMCIASocket slot;
20 PCMCIACardState *card;
Benoît Canet354a8c02011-10-30 14:50:12 +010021 MemoryRegion common_iomem;
Benoît Canet4beeaa72011-10-30 14:50:13 +010022 MemoryRegion attr_iomem;
Benoît Canet59aee132011-10-30 14:50:14 +010023 MemoryRegion iomem;
balroga171fe32007-04-30 01:48:07 +000024
25 qemu_irq irq;
26 qemu_irq cd_irq;
27};
28
Benoît Canet354a8c02011-10-30 14:50:12 +010029static uint64_t pxa2xx_pcmcia_common_read(void *opaque,
Avi Kivitya8170e52012-10-23 12:30:10 +020030 hwaddr offset, unsigned size)
balroga171fe32007-04-30 01:48:07 +000031{
Paul Brookbc24a222009-05-10 01:44:56 +010032 PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque;
balroga171fe32007-04-30 01:48:07 +000033
34 if (s->slot.attached) {
balroga171fe32007-04-30 01:48:07 +000035 return s->card->common_read(s->card->state, offset);
36 }
37
38 return 0;
39}
40
Avi Kivitya8170e52012-10-23 12:30:10 +020041static void pxa2xx_pcmcia_common_write(void *opaque, hwaddr offset,
Benoît Canet354a8c02011-10-30 14:50:12 +010042 uint64_t value, unsigned size)
balroga171fe32007-04-30 01:48:07 +000043{
Paul Brookbc24a222009-05-10 01:44:56 +010044 PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque;
balroga171fe32007-04-30 01:48:07 +000045
46 if (s->slot.attached) {
balroga171fe32007-04-30 01:48:07 +000047 s->card->common_write(s->card->state, offset, value);
48 }
49}
50
Benoît Canet4beeaa72011-10-30 14:50:13 +010051static uint64_t pxa2xx_pcmcia_attr_read(void *opaque,
Avi Kivitya8170e52012-10-23 12:30:10 +020052 hwaddr offset, unsigned size)
balroga171fe32007-04-30 01:48:07 +000053{
Paul Brookbc24a222009-05-10 01:44:56 +010054 PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque;
balroga171fe32007-04-30 01:48:07 +000055
56 if (s->slot.attached) {
balroga171fe32007-04-30 01:48:07 +000057 return s->card->attr_read(s->card->state, offset);
58 }
59
60 return 0;
61}
62
Avi Kivitya8170e52012-10-23 12:30:10 +020063static void pxa2xx_pcmcia_attr_write(void *opaque, hwaddr offset,
Benoît Canet4beeaa72011-10-30 14:50:13 +010064 uint64_t value, unsigned size)
balroga171fe32007-04-30 01:48:07 +000065{
Paul Brookbc24a222009-05-10 01:44:56 +010066 PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque;
balroga171fe32007-04-30 01:48:07 +000067
68 if (s->slot.attached) {
balroga171fe32007-04-30 01:48:07 +000069 s->card->attr_write(s->card->state, offset, value);
70 }
71}
72
Benoît Canet59aee132011-10-30 14:50:14 +010073static uint64_t pxa2xx_pcmcia_io_read(void *opaque,
Avi Kivitya8170e52012-10-23 12:30:10 +020074 hwaddr offset, unsigned size)
balroga171fe32007-04-30 01:48:07 +000075{
Paul Brookbc24a222009-05-10 01:44:56 +010076 PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque;
balroga171fe32007-04-30 01:48:07 +000077
78 if (s->slot.attached) {
balroga171fe32007-04-30 01:48:07 +000079 return s->card->io_read(s->card->state, offset);
80 }
81
82 return 0;
83}
84
Avi Kivitya8170e52012-10-23 12:30:10 +020085static void pxa2xx_pcmcia_io_write(void *opaque, hwaddr offset,
Benoît Canet59aee132011-10-30 14:50:14 +010086 uint64_t value, unsigned size)
balroga171fe32007-04-30 01:48:07 +000087{
Paul Brookbc24a222009-05-10 01:44:56 +010088 PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque;
balroga171fe32007-04-30 01:48:07 +000089
90 if (s->slot.attached) {
balroga171fe32007-04-30 01:48:07 +000091 s->card->io_write(s->card->state, offset, value);
92 }
93}
94
Benoît Canet354a8c02011-10-30 14:50:12 +010095static const MemoryRegionOps pxa2xx_pcmcia_common_ops = {
96 .read = pxa2xx_pcmcia_common_read,
97 .write = pxa2xx_pcmcia_common_write,
98 .endianness = DEVICE_NATIVE_ENDIAN
balroga171fe32007-04-30 01:48:07 +000099};
100
Benoît Canet4beeaa72011-10-30 14:50:13 +0100101static const MemoryRegionOps pxa2xx_pcmcia_attr_ops = {
102 .read = pxa2xx_pcmcia_attr_read,
103 .write = pxa2xx_pcmcia_attr_write,
104 .endianness = DEVICE_NATIVE_ENDIAN
balroga171fe32007-04-30 01:48:07 +0000105};
106
Benoît Canet59aee132011-10-30 14:50:14 +0100107static const MemoryRegionOps pxa2xx_pcmcia_io_ops = {
108 .read = pxa2xx_pcmcia_io_read,
109 .write = pxa2xx_pcmcia_io_write,
110 .endianness = DEVICE_NATIVE_ENDIAN
balroga171fe32007-04-30 01:48:07 +0000111};
112
113static void pxa2xx_pcmcia_set_irq(void *opaque, int line, int level)
114{
Paul Brookbc24a222009-05-10 01:44:56 +0100115 PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque;
balroga171fe32007-04-30 01:48:07 +0000116 if (!s->irq)
117 return;
118
119 qemu_set_irq(s->irq, level);
120}
121
Benoît Canet354a8c02011-10-30 14:50:12 +0100122PXA2xxPCMCIAState *pxa2xx_pcmcia_init(MemoryRegion *sysmem,
Avi Kivitya8170e52012-10-23 12:30:10 +0200123 hwaddr base)
balroga171fe32007-04-30 01:48:07 +0000124{
Paul Brookbc24a222009-05-10 01:44:56 +0100125 PXA2xxPCMCIAState *s;
balroga171fe32007-04-30 01:48:07 +0000126
Paul Brookbc24a222009-05-10 01:44:56 +0100127 s = (PXA2xxPCMCIAState *)
Anthony Liguori7267c092011-08-20 22:09:37 -0500128 g_malloc0(sizeof(PXA2xxPCMCIAState));
balroga171fe32007-04-30 01:48:07 +0000129
130 /* Socket I/O Memory Space */
Benoît Canet59aee132011-10-30 14:50:14 +0100131 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);
balroga171fe32007-04-30 01:48:07 +0000135
136 /* Then next 64 MB is reserved */
137
138 /* Socket Attribute Memory Space */
Benoît Canet4beeaa72011-10-30 14:50:13 +0100139 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);
balroga171fe32007-04-30 01:48:07 +0000143
144 /* Socket Common Memory Space */
Benoît Canet354a8c02011-10-30 14:50:12 +0100145 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);
balroga171fe32007-04-30 01:48:07 +0000149
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);
balrog3f582262007-05-23 21:47:51 +0000156
balroga171fe32007-04-30 01:48:07 +0000157 return s;
158}
159
160/* Insert a new card into a slot */
Paul Brookbc24a222009-05-10 01:44:56 +0100161int pxa2xx_pcmcia_attach(void *opaque, PCMCIACardState *card)
balroga171fe32007-04-30 01:48:07 +0000162{
Paul Brookbc24a222009-05-10 01:44:56 +0100163 PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque;
balroga171fe32007-04-30 01:48:07 +0000164 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 */
181int pxa2xx_pcmcia_dettach(void *opaque)
182{
Paul Brookbc24a222009-05-10 01:44:56 +0100183 PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque;
balroga171fe32007-04-30 01:48:07 +0000184 if (!s->slot.attached)
185 return -ENOENT;
186
187 s->card->detach(s->card->state);
Blue Swirlb9d38e92009-09-21 18:11:34 +0000188 s->card->slot = NULL;
189 s->card = NULL;
balroga171fe32007-04-30 01:48:07 +0000190
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 */
202void pxa2xx_pcmcia_set_irq_cb(void *opaque, qemu_irq irq, qemu_irq cd_irq)
203{
Paul Brookbc24a222009-05-10 01:44:56 +0100204 PXA2xxPCMCIAState *s = (PXA2xxPCMCIAState *) opaque;
balroga171fe32007-04-30 01:48:07 +0000205 s->irq = irq;
206 s->cd_irq = cd_irq;
207}