| /* |
| * Device's clock input and output |
| * |
| * Copyright GreenSocs 2016-2020 |
| * |
| * Authors: |
| * Frederic Konrad |
| * Damien Hedde |
| * |
| * This work is licensed under the terms of the GNU GPL, version 2 or later. |
| * See the COPYING file in the top-level directory. |
| */ |
| |
| #ifndef QDEV_CLOCK_H |
| #define QDEV_CLOCK_H |
| |
| #include "hw/clock.h" |
| |
| /** |
| * qdev_init_clock_in: |
| * @dev: the device to add an input clock to |
| * @name: the name of the clock (can't be NULL). |
| * @callback: optional callback to be called on update or NULL. |
| * @opaque: argument for the callback |
| * @events: the events the callback should be called for |
| * (logical OR of ClockEvent enum values) |
| * @returns: a pointer to the newly added clock |
| * |
| * Add an input clock to device @dev as a clock named @name. |
| * This adds a child<> property. |
| * The callback will be called with @opaque as opaque parameter. |
| */ |
| Clock *qdev_init_clock_in(DeviceState *dev, const char *name, |
| ClockCallback *callback, void *opaque, |
| unsigned int events); |
| |
| /** |
| * qdev_init_clock_out: |
| * @dev: the device to add an output clock to |
| * @name: the name of the clock (can't be NULL). |
| * @returns: a pointer to the newly added clock |
| * |
| * Add an output clock to device @dev as a clock named @name. |
| * This adds a child<> property. |
| */ |
| Clock *qdev_init_clock_out(DeviceState *dev, const char *name); |
| |
| /** |
| * qdev_get_clock_in: |
| * @dev: the device which has the clock |
| * @name: the name of the clock (can't be NULL). |
| * @returns: a pointer to the clock |
| * |
| * Get the input clock @name from @dev or NULL if does not exist. |
| */ |
| Clock *qdev_get_clock_in(DeviceState *dev, const char *name); |
| |
| /** |
| * qdev_get_clock_out: |
| * @dev: the device which has the clock |
| * @name: the name of the clock (can't be NULL). |
| * @returns: a pointer to the clock |
| * |
| * Get the output clock @name from @dev or NULL if does not exist. |
| */ |
| Clock *qdev_get_clock_out(DeviceState *dev, const char *name); |
| |
| /** |
| * qdev_connect_clock_in: |
| * @dev: a device |
| * @name: the name of an input clock in @dev |
| * @source: the source clock (an output clock of another device for example) |
| * |
| * Set the source clock of input clock @name of device @dev to @source. |
| * @source period update will be propagated to @name clock. |
| * |
| * Must be called before @dev is realized. |
| */ |
| void qdev_connect_clock_in(DeviceState *dev, const char *name, Clock *source); |
| |
| /** |
| * qdev_alias_clock: |
| * @dev: the device which has the clock |
| * @name: the name of the clock in @dev (can't be NULL) |
| * @alias_dev: the device to add the clock |
| * @alias_name: the name of the clock in @container |
| * @returns: a pointer to the clock |
| * |
| * Add a clock @alias_name in @alias_dev which is an alias of the clock @name |
| * in @dev. The direction _in_ or _out_ will the same as the original. |
| * An alias clock must not be modified or used by @alias_dev and should |
| * typically be only only for device composition purpose. |
| */ |
| Clock *qdev_alias_clock(DeviceState *dev, const char *name, |
| DeviceState *alias_dev, const char *alias_name); |
| |
| /** |
| * qdev_finalize_clocklist: |
| * @dev: the device being finalized |
| * |
| * Clear the clocklist from @dev. Only used internally in qdev. |
| */ |
| void qdev_finalize_clocklist(DeviceState *dev); |
| |
| /** |
| * ClockPortInitElem: |
| * @name: name of the clock (can't be NULL) |
| * @output: indicates whether the clock is input or output |
| * @callback: for inputs, optional callback to be called on clock's update |
| * with device as opaque |
| * @callback_events: mask of ClockEvent values for when callback is called |
| * @offset: optional offset to store the ClockIn or ClockOut pointer in device |
| * state structure (0 means unused) |
| */ |
| struct ClockPortInitElem { |
| const char *name; |
| bool is_output; |
| ClockCallback *callback; |
| unsigned int callback_events; |
| size_t offset; |
| }; |
| |
| #define clock_offset_value(devstate, field) \ |
| (offsetof(devstate, field) + \ |
| type_check(Clock *, typeof_field(devstate, field))) |
| |
| #define QDEV_CLOCK(out_not_in, devstate, field, cb, cbevents) { \ |
| .name = (stringify(field)), \ |
| .is_output = out_not_in, \ |
| .callback = cb, \ |
| .callback_events = cbevents, \ |
| .offset = clock_offset_value(devstate, field), \ |
| } |
| |
| /** |
| * QDEV_CLOCK_(IN|OUT): |
| * @devstate: structure type. @dev argument of qdev_init_clocks below must be |
| * a pointer to that same type. |
| * @field: a field in @_devstate (must be Clock*) |
| * @callback: (for input only) callback (or NULL) to be called with the device |
| * state as argument |
| * @cbevents: (for input only) ClockEvent mask for when callback is called |
| * |
| * The name of the clock will be derived from @field |
| */ |
| #define QDEV_CLOCK_IN(devstate, field, callback, cbevents) \ |
| QDEV_CLOCK(false, devstate, field, callback, cbevents) |
| |
| #define QDEV_CLOCK_OUT(devstate, field) \ |
| QDEV_CLOCK(true, devstate, field, NULL, 0) |
| |
| #define QDEV_CLOCK_END { .name = NULL } |
| |
| typedef struct ClockPortInitElem ClockPortInitArray[]; |
| |
| /** |
| * qdev_init_clocks: |
| * @dev: the device to add clocks to |
| * @clocks: a QDEV_CLOCK_END-terminated array which contains the |
| * clocks information. |
| */ |
| void qdev_init_clocks(DeviceState *dev, const ClockPortInitArray clocks); |
| |
| #endif /* QDEV_CLOCK_H */ |