]> git.proxmox.com Git - mirror_qemu.git/blame - include/hw/qdev-clock.h
Merge remote-tracking branch 'remotes/ericb/tags/pull-bitmaps-2020-09-21' into staging
[mirror_qemu.git] / include / hw / qdev-clock.h
CommitLineData
0e6934f2
DH
1/*
2 * Device's clock input and output
3 *
4 * Copyright GreenSocs 2016-2020
5 *
6 * Authors:
7 * Frederic Konrad
8 * Damien Hedde
9 *
10 * This work is licensed under the terms of the GNU GPL, version 2 or later.
11 * See the COPYING file in the top-level directory.
12 */
13
14#ifndef QDEV_CLOCK_H
15#define QDEV_CLOCK_H
16
17#include "hw/clock.h"
18
19/**
20 * qdev_init_clock_in:
21 * @dev: the device to add an input clock to
22 * @name: the name of the clock (can't be NULL).
23 * @callback: optional callback to be called on update or NULL.
24 * @opaque: argument for the callback
25 * @returns: a pointer to the newly added clock
26 *
27 * Add an input clock to device @dev as a clock named @name.
28 * This adds a child<> property.
29 * The callback will be called with @opaque as opaque parameter.
30 */
31Clock *qdev_init_clock_in(DeviceState *dev, const char *name,
32 ClockCallback *callback, void *opaque);
33
34/**
35 * qdev_init_clock_out:
36 * @dev: the device to add an output clock to
37 * @name: the name of the clock (can't be NULL).
38 * @returns: a pointer to the newly added clock
39 *
40 * Add an output clock to device @dev as a clock named @name.
41 * This adds a child<> property.
42 */
43Clock *qdev_init_clock_out(DeviceState *dev, const char *name);
44
45/**
46 * qdev_get_clock_in:
47 * @dev: the device which has the clock
48 * @name: the name of the clock (can't be NULL).
49 * @returns: a pointer to the clock
50 *
51 * Get the input clock @name from @dev or NULL if does not exist.
52 */
53Clock *qdev_get_clock_in(DeviceState *dev, const char *name);
54
55/**
56 * qdev_get_clock_out:
57 * @dev: the device which has the clock
58 * @name: the name of the clock (can't be NULL).
59 * @returns: a pointer to the clock
60 *
61 * Get the output clock @name from @dev or NULL if does not exist.
62 */
63Clock *qdev_get_clock_out(DeviceState *dev, const char *name);
64
65/**
66 * qdev_connect_clock_in:
67 * @dev: a device
68 * @name: the name of an input clock in @dev
69 * @source: the source clock (an output clock of another device for example)
70 *
71 * Set the source clock of input clock @name of device @dev to @source.
72 * @source period update will be propagated to @name clock.
739fa325
PMD
73 *
74 * Must be called before @dev is realized.
0e6934f2 75 */
f129360c 76void qdev_connect_clock_in(DeviceState *dev, const char *name, Clock *source);
0e6934f2
DH
77
78/**
79 * qdev_alias_clock:
80 * @dev: the device which has the clock
81 * @name: the name of the clock in @dev (can't be NULL)
82 * @alias_dev: the device to add the clock
83 * @alias_name: the name of the clock in @container
84 * @returns: a pointer to the clock
85 *
86 * Add a clock @alias_name in @alias_dev which is an alias of the clock @name
87 * in @dev. The direction _in_ or _out_ will the same as the original.
88 * An alias clock must not be modified or used by @alias_dev and should
89 * typically be only only for device composition purpose.
90 */
91Clock *qdev_alias_clock(DeviceState *dev, const char *name,
92 DeviceState *alias_dev, const char *alias_name);
93
94/**
95 * qdev_finalize_clocklist:
96 * @dev: the device being finalized
97 *
98 * Clear the clocklist from @dev. Only used internally in qdev.
99 */
100void qdev_finalize_clocklist(DeviceState *dev);
101
f0bc2a64
DH
102/**
103 * ClockPortInitElem:
104 * @name: name of the clock (can't be NULL)
105 * @output: indicates whether the clock is input or output
106 * @callback: for inputs, optional callback to be called on clock's update
107 * with device as opaque
108 * @offset: optional offset to store the ClockIn or ClockOut pointer in device
109 * state structure (0 means unused)
110 */
111struct ClockPortInitElem {
112 const char *name;
113 bool is_output;
114 ClockCallback *callback;
115 size_t offset;
116};
117
118#define clock_offset_value(devstate, field) \
119 (offsetof(devstate, field) + \
120 type_check(Clock *, typeof_field(devstate, field)))
121
122#define QDEV_CLOCK(out_not_in, devstate, field, cb) { \
123 .name = (stringify(field)), \
124 .is_output = out_not_in, \
125 .callback = cb, \
126 .offset = clock_offset_value(devstate, field), \
127}
128
129/**
130 * QDEV_CLOCK_(IN|OUT):
131 * @devstate: structure type. @dev argument of qdev_init_clocks below must be
132 * a pointer to that same type.
133 * @field: a field in @_devstate (must be Clock*)
134 * @callback: (for input only) callback (or NULL) to be called with the device
135 * state as argument
136 *
137 * The name of the clock will be derived from @field
138 */
139#define QDEV_CLOCK_IN(devstate, field, callback) \
140 QDEV_CLOCK(false, devstate, field, callback)
141
142#define QDEV_CLOCK_OUT(devstate, field) \
143 QDEV_CLOCK(true, devstate, field, NULL)
144
145#define QDEV_CLOCK_END { .name = NULL }
146
147typedef struct ClockPortInitElem ClockPortInitArray[];
148
149/**
150 * qdev_init_clocks:
151 * @dev: the device to add clocks to
152 * @clocks: a QDEV_CLOCK_END-terminated array which contains the
153 * clocks information.
154 */
155void qdev_init_clocks(DeviceState *dev, const ClockPortInitArray clocks);
156
0e6934f2 157#endif /* QDEV_CLOCK_H */