|  | /* | 
|  | * 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 */ |