blob: 3ac85216a5913b447ff7cef2a7fc29783b735f0e [file] [log] [blame]
Peter Maydell78c71af2016-06-14 15:59:15 +01001/* A simple I2C slave for returning monitor EDID data via DDC.
2 *
3 * Copyright (c) 2011 Linaro Limited
4 * Written by Peter Maydell
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License version 2 as
8 * published by the Free Software Foundation.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 *
15 * You should have received a copy of the GNU General Public License along
16 * with this program; if not, see <http://www.gnu.org/licenses/>.
17 */
18
19#include "qemu/osdep.h"
20#include "qemu/log.h"
Markus Armbruster0b8fa322019-05-23 16:35:07 +020021#include "qemu/module.h"
Peter Maydell78c71af2016-06-14 15:59:15 +010022#include "hw/i2c/i2c.h"
Markus Armbrusterd6454272019-08-12 07:23:45 +020023#include "migration/vmstate.h"
Paolo Bonzini6306cae2019-03-25 16:59:23 +010024#include "hw/display/i2c-ddc.h"
Peter Maydell78c71af2016-06-14 15:59:15 +010025
26#ifndef DEBUG_I2CDDC
27#define DEBUG_I2CDDC 0
28#endif
29
30#define DPRINTF(fmt, ...) do { \
31 if (DEBUG_I2CDDC) { \
32 qemu_log("i2c-ddc: " fmt , ## __VA_ARGS__); \
33 } \
Eric Blake25627552017-12-01 17:24:32 -060034} while (0)
Peter Maydell78c71af2016-06-14 15:59:15 +010035
Peter Maydell78c71af2016-06-14 15:59:15 +010036static void i2c_ddc_reset(DeviceState *ds)
37{
38 I2CDDCState *s = I2CDDC(ds);
39
40 s->firstbyte = false;
41 s->reg = 0;
42}
43
Corey Minyardd307c282017-01-09 11:40:20 +000044static int i2c_ddc_event(I2CSlave *i2c, enum i2c_event event)
Peter Maydell78c71af2016-06-14 15:59:15 +010045{
46 I2CDDCState *s = I2CDDC(i2c);
47
48 if (event == I2C_START_SEND) {
49 s->firstbyte = true;
50 }
Corey Minyardd307c282017-01-09 11:40:20 +000051
52 return 0;
Peter Maydell78c71af2016-06-14 15:59:15 +010053}
54
Corey Minyard2ac4c5f2018-11-14 11:50:50 -060055static uint8_t i2c_ddc_rx(I2CSlave *i2c)
Peter Maydell78c71af2016-06-14 15:59:15 +010056{
57 I2CDDCState *s = I2CDDC(i2c);
58
59 int value;
Gerd Hoffmannb05b2672019-01-08 11:23:01 +010060 value = s->edid_blob[s->reg % sizeof(s->edid_blob)];
Peter Maydell78c71af2016-06-14 15:59:15 +010061 s->reg++;
62 return value;
63}
64
65static int i2c_ddc_tx(I2CSlave *i2c, uint8_t data)
66{
67 I2CDDCState *s = I2CDDC(i2c);
68 if (s->firstbyte) {
69 s->reg = data;
70 s->firstbyte = false;
71 DPRINTF("[EDID] Written new pointer: %u\n", data);
Linus Walleij839a2b22018-03-01 11:05:45 +000072 return 0;
Peter Maydell78c71af2016-06-14 15:59:15 +010073 }
74
75 /* Ignore all writes */
76 s->reg++;
Linus Walleij839a2b22018-03-01 11:05:45 +000077 return 0;
Peter Maydell78c71af2016-06-14 15:59:15 +010078}
79
80static void i2c_ddc_init(Object *obj)
81{
82 I2CDDCState *s = I2CDDC(obj);
Gerd Hoffmann715eb052018-10-05 13:08:37 +020083
84 qemu_edid_generate(s->edid_blob, sizeof(s->edid_blob), &s->edid_info);
Peter Maydell78c71af2016-06-14 15:59:15 +010085}
86
87static const VMStateDescription vmstate_i2c_ddc = {
88 .name = TYPE_I2CDDC,
89 .version_id = 1,
90 .fields = (VMStateField[]) {
91 VMSTATE_BOOL(firstbyte, I2CDDCState),
92 VMSTATE_UINT8(reg, I2CDDCState),
93 VMSTATE_END_OF_LIST()
94 }
95};
96
Gerd Hoffmann715eb052018-10-05 13:08:37 +020097static Property i2c_ddc_properties[] = {
98 DEFINE_EDID_PROPERTIES(I2CDDCState, edid_info),
99 DEFINE_PROP_END_OF_LIST(),
100};
101
Peter Maydell78c71af2016-06-14 15:59:15 +0100102static void i2c_ddc_class_init(ObjectClass *oc, void *data)
103{
104 DeviceClass *dc = DEVICE_CLASS(oc);
105 I2CSlaveClass *isc = I2C_SLAVE_CLASS(oc);
106
107 dc->reset = i2c_ddc_reset;
108 dc->vmsd = &vmstate_i2c_ddc;
Gerd Hoffmann715eb052018-10-05 13:08:37 +0200109 dc->props = i2c_ddc_properties;
Peter Maydell78c71af2016-06-14 15:59:15 +0100110 isc->event = i2c_ddc_event;
111 isc->recv = i2c_ddc_rx;
112 isc->send = i2c_ddc_tx;
113}
114
115static TypeInfo i2c_ddc_info = {
116 .name = TYPE_I2CDDC,
117 .parent = TYPE_I2C_SLAVE,
118 .instance_size = sizeof(I2CDDCState),
119 .instance_init = i2c_ddc_init,
120 .class_init = i2c_ddc_class_init
121};
122
123static void ddc_register_devices(void)
124{
125 type_register_static(&i2c_ddc_info);
126}
127
128type_init(ddc_register_devices);