qemu/include/hw/qdev-clock.h
Damien Hedde f0bc2a64c0 qdev-clock: introduce an init array to ease the device construction
Introduce a function and macro helpers to setup several clocks
in a device from a static array description.

An element of the array describes the clock (name and direction) as
well as the related callback and an optional offset to store the
created object pointer in the device state structure.

The array must be terminated by a special element QDEV_CLOCK_END.

This is based on the original work of Frederic Konrad.

Signed-off-by: Damien Hedde <damien.hedde@greensocs.com>
Reviewed-by: Philippe Mathieu-Daudé <philmd@redhat.com>
Reviewed-by: Alistair Francis <alistair.francis@wdc.com>
Reviewed-by: Edgar E. Iglesias <edgar.iglesias@xilinx.com>
Message-id: 20200406135251.157596-5-damien.hedde@greensocs.com
Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
2020-04-30 15:35:40 +01:00

160 lines
4.8 KiB
C

/*
* 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
* @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);
/**
* 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.
*/
static inline void qdev_connect_clock_in(DeviceState *dev, const char *name,
Clock *source)
{
clock_set_source(qdev_get_clock_in(dev, name), 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
* @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;
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) { \
.name = (stringify(field)), \
.is_output = out_not_in, \
.callback = cb, \
.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
*
* The name of the clock will be derived from @field
*/
#define QDEV_CLOCK_IN(devstate, field, callback) \
QDEV_CLOCK(false, devstate, field, callback)
#define QDEV_CLOCK_OUT(devstate, field) \
QDEV_CLOCK(true, devstate, field, NULL)
#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 */