~ubuntu-branches/ubuntu/trusty/qemu/trusty

« back to all changes in this revision

Viewing changes to .pc/ubuntu/linaro/0028-hw-omap_spi.c-prepare-for-omap3.patch/hw/ssi/omap_spi.c

  • Committer: Package Import Robot
  • Author(s): Serge Hallyn
  • Date: 2014-02-04 12:13:08 UTC
  • mfrom: (10.1.45 sid)
  • Revision ID: package-import@ubuntu.com-20140204121308-1xq92lrfs75agw2g
Tags: 1.7.0+dfsg-3ubuntu1~ppa1
* Merge 1.7.0+dfsg-3 from debian.  Remaining changes:
  - debian/patches/ubuntu:
    * expose-vmx_qemu64cpu.patch
    * linaro (omap3) and arm64 patches
    * ubuntu/target-ppc-add-stubs-for-kvm-breakpoints: fix FTBFS
      on ppc
    * ubuntu/CVE-2013-4377.patch: fix denial of service via virtio
  - debian/qemu-system-x86.modprobe: set kvm_intel nested=1 options
  - debian/control:
    * add arm64 to Architectures
    * add qemu-common and qemu-system-aarch64 packages
  - debian/qemu-system-common.install: add debian/tmp/usr/lib
  - debian/qemu-system-common.preinst: add kvm group
  - debian/qemu-system-common.postinst: remove acl placed by udev,
    and add udevadm trigger.
  - qemu-system-x86.links: add eepro100.rom, remove pxe-virtio,
    pxe-e1000 and pxe-rtl8139.
  - add qemu-system-x86.qemu-kvm.upstart and .default
  - qemu-user-static.postinst-in: remove arm64 binfmt
  - debian/rules:
    * allow parallel build
    * add aarch64 to system_targets and sys_systems
    * add qemu-kvm-spice links
    * install qemu-system-x86.modprobe
  - add debian/qemu-system-common.links for OVMF.fd link
* Remove kvm-img, kvm-nbd, kvm-ifup and kvm-ifdown symlinks.

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
/*
 
2
 * TI OMAP processor's Multichannel SPI emulation.
 
3
 *
 
4
 * Copyright (C) 2007-2009 Nokia Corporation
 
5
 *
 
6
 * Original code for OMAP2 by Andrzej Zaborowski <andrew@openedhand.com>
 
7
 *
 
8
 * This program is free software; you can redistribute it and/or
 
9
 * modify it under the terms of the GNU General Public License as
 
10
 * published by the Free Software Foundation; either version 2 or
 
11
 * (at your option) any later version of the License.
 
12
 *
 
13
 * This program is distributed in the hope that it will be useful,
 
14
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 
15
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
16
 * GNU General Public License for more details.
 
17
 *
 
18
 * You should have received a copy of the GNU General Public License along
 
19
 * with this program; if not, write to the Free Software Foundation, Inc.,
 
20
 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
 
21
 */
 
22
#include "hw/hw.h"
 
23
#include "hw/arm/omap.h"
 
24
 
 
25
/* Multichannel SPI */
 
26
struct omap_mcspi_s {
 
27
    MemoryRegion iomem;
 
28
    qemu_irq irq;
 
29
    int chnum;
 
30
 
 
31
    uint32_t sysconfig;
 
32
    uint32_t systest;
 
33
    uint32_t irqst;
 
34
    uint32_t irqen;
 
35
    uint32_t wken;
 
36
    uint32_t control;
 
37
 
 
38
    struct omap_mcspi_ch_s {
 
39
        qemu_irq txdrq;
 
40
        qemu_irq rxdrq;
 
41
        uint32_t (*txrx)(void *opaque, uint32_t, int);
 
42
        void *opaque;
 
43
 
 
44
        uint32_t tx;
 
45
        uint32_t rx;
 
46
 
 
47
        uint32_t config;
 
48
        uint32_t status;
 
49
        uint32_t control;
 
50
    } ch[4];
 
51
};
 
52
 
 
53
static inline void omap_mcspi_interrupt_update(struct omap_mcspi_s *s)
 
54
{
 
55
    qemu_set_irq(s->irq, s->irqst & s->irqen);
 
56
}
 
57
 
 
58
static inline void omap_mcspi_dmarequest_update(struct omap_mcspi_ch_s *ch)
 
59
{
 
60
    qemu_set_irq(ch->txdrq,
 
61
                    (ch->control & 1) &&                /* EN */
 
62
                    (ch->config & (1 << 14)) &&         /* DMAW */
 
63
                    (ch->status & (1 << 1)) &&          /* TXS */
 
64
                    ((ch->config >> 12) & 3) != 1);     /* TRM */
 
65
    qemu_set_irq(ch->rxdrq,
 
66
                    (ch->control & 1) &&                /* EN */
 
67
                    (ch->config & (1 << 15)) &&         /* DMAW */
 
68
                    (ch->status & (1 << 0)) &&          /* RXS */
 
69
                    ((ch->config >> 12) & 3) != 2);     /* TRM */
 
70
}
 
71
 
 
72
static void omap_mcspi_transfer_run(struct omap_mcspi_s *s, int chnum)
 
73
{
 
74
    struct omap_mcspi_ch_s *ch = s->ch + chnum;
 
75
 
 
76
    if (!(ch->control & 1))                             /* EN */
 
77
        return;
 
78
    if ((ch->status & (1 << 0)) &&                      /* RXS */
 
79
                    ((ch->config >> 12) & 3) != 2 &&    /* TRM */
 
80
                    !(ch->config & (1 << 19)))          /* TURBO */
 
81
        goto intr_update;
 
82
    if ((ch->status & (1 << 1)) &&                      /* TXS */
 
83
                    ((ch->config >> 12) & 3) != 1)      /* TRM */
 
84
        goto intr_update;
 
85
 
 
86
    if (!(s->control & 1) ||                            /* SINGLE */
 
87
                    (ch->config & (1 << 20))) {         /* FORCE */
 
88
        if (ch->txrx)
 
89
            ch->rx = ch->txrx(ch->opaque, ch->tx,       /* WL */
 
90
                            1 + (0x1f & (ch->config >> 7)));
 
91
    }
 
92
 
 
93
    ch->tx = 0;
 
94
    ch->status |= 1 << 2;                               /* EOT */
 
95
    ch->status |= 1 << 1;                               /* TXS */
 
96
    if (((ch->config >> 12) & 3) != 2)                  /* TRM */
 
97
        ch->status |= 1 << 0;                           /* RXS */
 
98
 
 
99
intr_update:
 
100
    if ((ch->status & (1 << 0)) &&                      /* RXS */
 
101
                    ((ch->config >> 12) & 3) != 2 &&    /* TRM */
 
102
                    !(ch->config & (1 << 19)))          /* TURBO */
 
103
        s->irqst |= 1 << (2 + 4 * chnum);               /* RX_FULL */
 
104
    if ((ch->status & (1 << 1)) &&                      /* TXS */
 
105
                    ((ch->config >> 12) & 3) != 1)      /* TRM */
 
106
        s->irqst |= 1 << (0 + 4 * chnum);               /* TX_EMPTY */
 
107
    omap_mcspi_interrupt_update(s);
 
108
    omap_mcspi_dmarequest_update(ch);
 
109
}
 
110
 
 
111
void omap_mcspi_reset(struct omap_mcspi_s *s)
 
112
{
 
113
    int ch;
 
114
 
 
115
    s->sysconfig = 0;
 
116
    s->systest = 0;
 
117
    s->irqst = 0;
 
118
    s->irqen = 0;
 
119
    s->wken = 0;
 
120
    s->control = 4;
 
121
 
 
122
    for (ch = 0; ch < 4; ch ++) {
 
123
        s->ch[ch].config = 0x060000;
 
124
        s->ch[ch].status = 2;                           /* TXS */
 
125
        s->ch[ch].control = 0;
 
126
 
 
127
        omap_mcspi_dmarequest_update(s->ch + ch);
 
128
    }
 
129
 
 
130
    omap_mcspi_interrupt_update(s);
 
131
}
 
132
 
 
133
static uint64_t omap_mcspi_read(void *opaque, hwaddr addr,
 
134
                                unsigned size)
 
135
{
 
136
    struct omap_mcspi_s *s = (struct omap_mcspi_s *) opaque;
 
137
    int ch = 0;
 
138
    uint32_t ret;
 
139
 
 
140
    if (size != 4) {
 
141
        return omap_badwidth_read32(opaque, addr);
 
142
    }
 
143
 
 
144
    switch (addr) {
 
145
    case 0x00:  /* MCSPI_REVISION */
 
146
        return 0x91;
 
147
 
 
148
    case 0x10:  /* MCSPI_SYSCONFIG */
 
149
        return s->sysconfig;
 
150
 
 
151
    case 0x14:  /* MCSPI_SYSSTATUS */
 
152
        return 1;                                       /* RESETDONE */
 
153
 
 
154
    case 0x18:  /* MCSPI_IRQSTATUS */
 
155
        return s->irqst;
 
156
 
 
157
    case 0x1c:  /* MCSPI_IRQENABLE */
 
158
        return s->irqen;
 
159
 
 
160
    case 0x20:  /* MCSPI_WAKEUPENABLE */
 
161
        return s->wken;
 
162
 
 
163
    case 0x24:  /* MCSPI_SYST */
 
164
        return s->systest;
 
165
 
 
166
    case 0x28:  /* MCSPI_MODULCTRL */
 
167
        return s->control;
 
168
 
 
169
    case 0x68: ch ++;
 
170
        /* fall through */
 
171
    case 0x54: ch ++;
 
172
        /* fall through */
 
173
    case 0x40: ch ++;
 
174
        /* fall through */
 
175
    case 0x2c:  /* MCSPI_CHCONF */
 
176
        return s->ch[ch].config;
 
177
 
 
178
    case 0x6c: ch ++;
 
179
        /* fall through */
 
180
    case 0x58: ch ++;
 
181
        /* fall through */
 
182
    case 0x44: ch ++;
 
183
        /* fall through */
 
184
    case 0x30:  /* MCSPI_CHSTAT */
 
185
        return s->ch[ch].status;
 
186
 
 
187
    case 0x70: ch ++;
 
188
        /* fall through */
 
189
    case 0x5c: ch ++;
 
190
        /* fall through */
 
191
    case 0x48: ch ++;
 
192
        /* fall through */
 
193
    case 0x34:  /* MCSPI_CHCTRL */
 
194
        return s->ch[ch].control;
 
195
 
 
196
    case 0x74: ch ++;
 
197
        /* fall through */
 
198
    case 0x60: ch ++;
 
199
        /* fall through */
 
200
    case 0x4c: ch ++;
 
201
        /* fall through */
 
202
    case 0x38:  /* MCSPI_TX */
 
203
        return s->ch[ch].tx;
 
204
 
 
205
    case 0x78: ch ++;
 
206
        /* fall through */
 
207
    case 0x64: ch ++;
 
208
        /* fall through */
 
209
    case 0x50: ch ++;
 
210
        /* fall through */
 
211
    case 0x3c:  /* MCSPI_RX */
 
212
        s->ch[ch].status &= ~(1 << 0);                  /* RXS */
 
213
        ret = s->ch[ch].rx;
 
214
        omap_mcspi_transfer_run(s, ch);
 
215
        return ret;
 
216
    }
 
217
 
 
218
    OMAP_BAD_REG(addr);
 
219
    return 0;
 
220
}
 
221
 
 
222
static void omap_mcspi_write(void *opaque, hwaddr addr,
 
223
                             uint64_t value, unsigned size)
 
224
{
 
225
    struct omap_mcspi_s *s = (struct omap_mcspi_s *) opaque;
 
226
    int ch = 0;
 
227
 
 
228
    if (size != 4) {
 
229
        return omap_badwidth_write32(opaque, addr, value);
 
230
    }
 
231
 
 
232
    switch (addr) {
 
233
    case 0x00:  /* MCSPI_REVISION */
 
234
    case 0x14:  /* MCSPI_SYSSTATUS */
 
235
    case 0x30:  /* MCSPI_CHSTAT0 */
 
236
    case 0x3c:  /* MCSPI_RX0 */
 
237
    case 0x44:  /* MCSPI_CHSTAT1 */
 
238
    case 0x50:  /* MCSPI_RX1 */
 
239
    case 0x58:  /* MCSPI_CHSTAT2 */
 
240
    case 0x64:  /* MCSPI_RX2 */
 
241
    case 0x6c:  /* MCSPI_CHSTAT3 */
 
242
    case 0x78:  /* MCSPI_RX3 */
 
243
        OMAP_RO_REG(addr);
 
244
        return;
 
245
 
 
246
    case 0x10:  /* MCSPI_SYSCONFIG */
 
247
        if (value & (1 << 1))                           /* SOFTRESET */
 
248
            omap_mcspi_reset(s);
 
249
        s->sysconfig = value & 0x31d;
 
250
        break;
 
251
 
 
252
    case 0x18:  /* MCSPI_IRQSTATUS */
 
253
        if (!((s->control & (1 << 3)) && (s->systest & (1 << 11)))) {
 
254
            s->irqst &= ~value;
 
255
            omap_mcspi_interrupt_update(s);
 
256
        }
 
257
        break;
 
258
 
 
259
    case 0x1c:  /* MCSPI_IRQENABLE */
 
260
        s->irqen = value & 0x1777f;
 
261
        omap_mcspi_interrupt_update(s);
 
262
        break;
 
263
 
 
264
    case 0x20:  /* MCSPI_WAKEUPENABLE */
 
265
        s->wken = value & 1;
 
266
        break;
 
267
 
 
268
    case 0x24:  /* MCSPI_SYST */
 
269
        if (s->control & (1 << 3))                      /* SYSTEM_TEST */
 
270
            if (value & (1 << 11)) {                    /* SSB */
 
271
                s->irqst |= 0x1777f;
 
272
                omap_mcspi_interrupt_update(s);
 
273
            }
 
274
        s->systest = value & 0xfff;
 
275
        break;
 
276
 
 
277
    case 0x28:  /* MCSPI_MODULCTRL */
 
278
        if (value & (1 << 3))                           /* SYSTEM_TEST */
 
279
            if (s->systest & (1 << 11)) {               /* SSB */
 
280
                s->irqst |= 0x1777f;
 
281
                omap_mcspi_interrupt_update(s);
 
282
            }
 
283
        s->control = value & 0xf;
 
284
        break;
 
285
 
 
286
    case 0x68: ch ++;
 
287
        /* fall through */
 
288
    case 0x54: ch ++;
 
289
        /* fall through */
 
290
    case 0x40: ch ++;
 
291
        /* fall through */
 
292
    case 0x2c:  /* MCSPI_CHCONF */
 
293
        if ((value ^ s->ch[ch].config) & (3 << 14))     /* DMAR | DMAW */
 
294
            omap_mcspi_dmarequest_update(s->ch + ch);
 
295
        if (((value >> 12) & 3) == 3)                   /* TRM */
 
296
            fprintf(stderr, "%s: invalid TRM value (3)\n", __FUNCTION__);
 
297
        if (((value >> 7) & 0x1f) < 3)                  /* WL */
 
298
            fprintf(stderr, "%s: invalid WL value (%" PRIx64 ")\n",
 
299
                            __FUNCTION__, (value >> 7) & 0x1f);
 
300
        s->ch[ch].config = value & 0x7fffff;
 
301
        break;
 
302
 
 
303
    case 0x70: ch ++;
 
304
        /* fall through */
 
305
    case 0x5c: ch ++;
 
306
        /* fall through */
 
307
    case 0x48: ch ++;
 
308
        /* fall through */
 
309
    case 0x34:  /* MCSPI_CHCTRL */
 
310
        if (value & ~s->ch[ch].control & 1) {           /* EN */
 
311
            s->ch[ch].control |= 1;
 
312
            omap_mcspi_transfer_run(s, ch);
 
313
        } else
 
314
            s->ch[ch].control = value & 1;
 
315
        break;
 
316
 
 
317
    case 0x74: ch ++;
 
318
        /* fall through */
 
319
    case 0x60: ch ++;
 
320
        /* fall through */
 
321
    case 0x4c: ch ++;
 
322
        /* fall through */
 
323
    case 0x38:  /* MCSPI_TX */
 
324
        s->ch[ch].tx = value;
 
325
        s->ch[ch].status &= ~(1 << 1);                  /* TXS */
 
326
        omap_mcspi_transfer_run(s, ch);
 
327
        break;
 
328
 
 
329
    default:
 
330
        OMAP_BAD_REG(addr);
 
331
        return;
 
332
    }
 
333
}
 
334
 
 
335
static const MemoryRegionOps omap_mcspi_ops = {
 
336
    .read = omap_mcspi_read,
 
337
    .write = omap_mcspi_write,
 
338
    .endianness = DEVICE_NATIVE_ENDIAN,
 
339
};
 
340
 
 
341
struct omap_mcspi_s *omap_mcspi_init(struct omap_target_agent_s *ta, int chnum,
 
342
                qemu_irq irq, qemu_irq *drq, omap_clk fclk, omap_clk iclk)
 
343
{
 
344
    struct omap_mcspi_s *s = (struct omap_mcspi_s *)
 
345
            g_malloc0(sizeof(struct omap_mcspi_s));
 
346
    struct omap_mcspi_ch_s *ch = s->ch;
 
347
 
 
348
    s->irq = irq;
 
349
    s->chnum = chnum;
 
350
    while (chnum --) {
 
351
        ch->txdrq = *drq ++;
 
352
        ch->rxdrq = *drq ++;
 
353
        ch ++;
 
354
    }
 
355
    omap_mcspi_reset(s);
 
356
 
 
357
    memory_region_init_io(&s->iomem, NULL, &omap_mcspi_ops, s, "omap.mcspi",
 
358
                          omap_l4_region_size(ta, 0));
 
359
    omap_l4_attach(ta, 0, &s->iomem);
 
360
 
 
361
    return s;
 
362
}
 
363
 
 
364
void omap_mcspi_attach(struct omap_mcspi_s *s,
 
365
                uint32_t (*txrx)(void *opaque, uint32_t, int), void *opaque,
 
366
                int chipselect)
 
367
{
 
368
    if (chipselect < 0 || chipselect >= s->chnum)
 
369
        hw_error("%s: Bad chipselect %i\n", __FUNCTION__, chipselect);
 
370
 
 
371
    s->ch[chipselect].txrx = txrx;
 
372
    s->ch[chipselect].opaque = opaque;
 
373
}